summaryrefslogtreecommitdiff
path: root/op_crates/webgpu/render_pass.rs
diff options
context:
space:
mode:
Diffstat (limited to 'op_crates/webgpu/render_pass.rs')
-rw-r--r--op_crates/webgpu/render_pass.rs676
1 files changed, 676 insertions, 0 deletions
diff --git a/op_crates/webgpu/render_pass.rs b/op_crates/webgpu/render_pass.rs
new file mode 100644
index 000000000..c9018726b
--- /dev/null
+++ b/op_crates/webgpu/render_pass.rs
@@ -0,0 +1,676 @@
+// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.
+
+use deno_core::error::bad_resource_id;
+use deno_core::error::AnyError;
+use deno_core::serde_json::json;
+use deno_core::serde_json::Value;
+use deno_core::ZeroCopyBuf;
+use deno_core::{OpState, Resource};
+use serde::Deserialize;
+use std::borrow::Cow;
+use std::cell::RefCell;
+
+use super::error::WebGPUError;
+
+pub(crate) struct WebGPURenderPass(
+ pub(crate) RefCell<wgpu_core::command::RenderPass>,
+);
+impl Resource for WebGPURenderPass {
+ fn name(&self) -> Cow<str> {
+ "webGPURenderPass".into()
+ }
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassSetViewportArgs {
+ render_pass_rid: u32,
+ x: f32,
+ y: f32,
+ width: f32,
+ height: f32,
+ min_depth: f32,
+ max_depth: f32,
+}
+
+pub fn op_webgpu_render_pass_set_viewport(
+ state: &mut OpState,
+ args: RenderPassSetViewportArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ wgpu_core::command::render_ffi::wgpu_render_pass_set_viewport(
+ &mut render_pass_resource.0.borrow_mut(),
+ args.x,
+ args.y,
+ args.width,
+ args.height,
+ args.min_depth,
+ args.max_depth,
+ );
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassSetScissorRectArgs {
+ render_pass_rid: u32,
+ x: u32,
+ y: u32,
+ width: u32,
+ height: u32,
+}
+
+pub fn op_webgpu_render_pass_set_scissor_rect(
+ state: &mut OpState,
+ args: RenderPassSetScissorRectArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ wgpu_core::command::render_ffi::wgpu_render_pass_set_scissor_rect(
+ &mut render_pass_resource.0.borrow_mut(),
+ args.x,
+ args.y,
+ args.width,
+ args.height,
+ );
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct GPUColor {
+ pub r: f64,
+ pub g: f64,
+ pub b: f64,
+ pub a: f64,
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassSetBlendColorArgs {
+ render_pass_rid: u32,
+ color: GPUColor,
+}
+
+pub fn op_webgpu_render_pass_set_blend_color(
+ state: &mut OpState,
+ args: RenderPassSetBlendColorArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ wgpu_core::command::render_ffi::wgpu_render_pass_set_blend_color(
+ &mut render_pass_resource.0.borrow_mut(),
+ &wgpu_types::Color {
+ r: args.color.r,
+ g: args.color.g,
+ b: args.color.b,
+ a: args.color.a,
+ },
+ );
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassSetStencilReferenceArgs {
+ render_pass_rid: u32,
+ reference: u32,
+}
+
+pub fn op_webgpu_render_pass_set_stencil_reference(
+ state: &mut OpState,
+ args: RenderPassSetStencilReferenceArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ wgpu_core::command::render_ffi::wgpu_render_pass_set_stencil_reference(
+ &mut render_pass_resource.0.borrow_mut(),
+ args.reference,
+ );
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassBeginPipelineStatisticsQueryArgs {
+ render_pass_rid: u32,
+ query_set: u32,
+ query_index: u32,
+}
+
+pub fn op_webgpu_render_pass_begin_pipeline_statistics_query(
+ state: &mut OpState,
+ args: RenderPassBeginPipelineStatisticsQueryArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+ let query_set_resource = state
+ .resource_table
+ .get::<super::WebGPUQuerySet>(args.query_set)
+ .ok_or_else(bad_resource_id)?;
+
+ unsafe {
+ wgpu_core::command::render_ffi::wgpu_render_pass_begin_pipeline_statistics_query(
+ &mut render_pass_resource.0.borrow_mut(),
+ query_set_resource.0,
+ args.query_index,
+ );
+ }
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassEndPipelineStatisticsQueryArgs {
+ render_pass_rid: u32,
+}
+
+pub fn op_webgpu_render_pass_end_pipeline_statistics_query(
+ state: &mut OpState,
+ args: RenderPassEndPipelineStatisticsQueryArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ unsafe {
+ wgpu_core::command::render_ffi::wgpu_render_pass_end_pipeline_statistics_query(
+ &mut render_pass_resource.0.borrow_mut(),
+ );
+ }
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassWriteTimestampArgs {
+ render_pass_rid: u32,
+ query_set: u32,
+ query_index: u32,
+}
+
+pub fn op_webgpu_render_pass_write_timestamp(
+ state: &mut OpState,
+ args: RenderPassWriteTimestampArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+ let query_set_resource = state
+ .resource_table
+ .get::<super::WebGPUQuerySet>(args.query_set)
+ .ok_or_else(bad_resource_id)?;
+
+ unsafe {
+ wgpu_core::command::render_ffi::wgpu_render_pass_write_timestamp(
+ &mut render_pass_resource.0.borrow_mut(),
+ query_set_resource.0,
+ args.query_index,
+ );
+ }
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassExecuteBundlesArgs {
+ render_pass_rid: u32,
+ bundles: Vec<u32>,
+}
+
+pub fn op_webgpu_render_pass_execute_bundles(
+ state: &mut OpState,
+ args: RenderPassExecuteBundlesArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let mut render_bundle_ids = vec![];
+
+ for rid in &args.bundles {
+ let render_bundle_resource = state
+ .resource_table
+ .get::<super::bundle::WebGPURenderBundle>(*rid)
+ .ok_or_else(bad_resource_id)?;
+ render_bundle_ids.push(render_bundle_resource.0);
+ }
+
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ unsafe {
+ wgpu_core::command::render_ffi::wgpu_render_pass_execute_bundles(
+ &mut render_pass_resource.0.borrow_mut(),
+ render_bundle_ids.as_ptr(),
+ args.bundles.len(),
+ );
+ }
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassEndPassArgs {
+ command_encoder_rid: u32,
+ render_pass_rid: u32,
+}
+
+pub fn op_webgpu_render_pass_end_pass(
+ state: &mut OpState,
+ args: RenderPassEndPassArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let command_encoder_resource = state
+ .resource_table
+ .get::<super::command_encoder::WebGPUCommandEncoder>(
+ args.command_encoder_rid,
+ )
+ .ok_or_else(bad_resource_id)?;
+ let command_encoder = command_encoder_resource.0;
+ let render_pass_resource = state
+ .resource_table
+ .take::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+ let render_pass = &render_pass_resource.0.borrow();
+ let instance = state.borrow::<super::Instance>();
+
+ let maybe_err = gfx_select!(command_encoder => instance.command_encoder_run_render_pass(command_encoder, render_pass)).err();
+
+ Ok(json!({ "err": maybe_err.map(WebGPUError::from) }))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassSetBindGroupArgs {
+ render_pass_rid: u32,
+ index: u32,
+ bind_group: u32,
+ dynamic_offsets_data: Option<Vec<u32>>,
+ dynamic_offsets_data_start: usize,
+ dynamic_offsets_data_length: usize,
+}
+
+pub fn op_webgpu_render_pass_set_bind_group(
+ state: &mut OpState,
+ args: RenderPassSetBindGroupArgs,
+ zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let bind_group_resource = state
+ .resource_table
+ .get::<super::binding::WebGPUBindGroup>(args.bind_group)
+ .ok_or_else(bad_resource_id)?;
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ // I know this might look like it can be easily deduplicated, but it can not
+ // be due to the lifetime of the args.dynamic_offsets_data slice. Because we
+ // need to use a raw pointer here the slice can be freed before the pointer
+ // is used in wgpu_render_pass_set_bind_group. See
+ // https://matrix.to/#/!XFRnMvAfptAHthwBCx:matrix.org/$HgrlhD-Me1DwsGb8UdMu2Hqubgks8s7ILwWRwigOUAg
+ match args.dynamic_offsets_data {
+ Some(data) => unsafe {
+ wgpu_core::command::render_ffi::wgpu_render_pass_set_bind_group(
+ &mut render_pass_resource.0.borrow_mut(),
+ args.index,
+ bind_group_resource.0,
+ data.as_slice().as_ptr(),
+ args.dynamic_offsets_data_length,
+ );
+ },
+ None => {
+ let (prefix, data, suffix) = unsafe { zero_copy[0].align_to::<u32>() };
+ assert!(prefix.is_empty());
+ assert!(suffix.is_empty());
+ unsafe {
+ wgpu_core::command::render_ffi::wgpu_render_pass_set_bind_group(
+ &mut render_pass_resource.0.borrow_mut(),
+ args.index,
+ bind_group_resource.0,
+ data[args.dynamic_offsets_data_start..].as_ptr(),
+ args.dynamic_offsets_data_length,
+ );
+ }
+ }
+ };
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassPushDebugGroupArgs {
+ render_pass_rid: u32,
+ group_label: String,
+}
+
+pub fn op_webgpu_render_pass_push_debug_group(
+ state: &mut OpState,
+ args: RenderPassPushDebugGroupArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ unsafe {
+ let label = std::ffi::CString::new(args.group_label).unwrap();
+ wgpu_core::command::render_ffi::wgpu_render_pass_push_debug_group(
+ &mut render_pass_resource.0.borrow_mut(),
+ label.as_ptr(),
+ 0, // wgpu#975
+ );
+ }
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassPopDebugGroupArgs {
+ render_pass_rid: u32,
+}
+
+pub fn op_webgpu_render_pass_pop_debug_group(
+ state: &mut OpState,
+ args: RenderPassPopDebugGroupArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ wgpu_core::command::render_ffi::wgpu_render_pass_pop_debug_group(
+ &mut render_pass_resource.0.borrow_mut(),
+ );
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassInsertDebugMarkerArgs {
+ render_pass_rid: u32,
+ marker_label: String,
+}
+
+pub fn op_webgpu_render_pass_insert_debug_marker(
+ state: &mut OpState,
+ args: RenderPassInsertDebugMarkerArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ unsafe {
+ let label = std::ffi::CString::new(args.marker_label).unwrap();
+ wgpu_core::command::render_ffi::wgpu_render_pass_insert_debug_marker(
+ &mut render_pass_resource.0.borrow_mut(),
+ label.as_ptr(),
+ 0, // wgpu#975
+ );
+ }
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassSetPipelineArgs {
+ render_pass_rid: u32,
+ pipeline: u32,
+}
+
+pub fn op_webgpu_render_pass_set_pipeline(
+ state: &mut OpState,
+ args: RenderPassSetPipelineArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pipeline_resource = state
+ .resource_table
+ .get::<super::pipeline::WebGPURenderPipeline>(args.pipeline)
+ .ok_or_else(bad_resource_id)?;
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ wgpu_core::command::render_ffi::wgpu_render_pass_set_pipeline(
+ &mut render_pass_resource.0.borrow_mut(),
+ render_pipeline_resource.0,
+ );
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassSetIndexBufferArgs {
+ render_pass_rid: u32,
+ buffer: u32,
+ index_format: String,
+ offset: u64,
+ size: u64,
+}
+
+pub fn op_webgpu_render_pass_set_index_buffer(
+ state: &mut OpState,
+ args: RenderPassSetIndexBufferArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let buffer_resource = state
+ .resource_table
+ .get::<super::buffer::WebGPUBuffer>(args.buffer)
+ .ok_or_else(bad_resource_id)?;
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ render_pass_resource.0.borrow_mut().set_index_buffer(
+ buffer_resource.0,
+ super::pipeline::serialize_index_format(args.index_format),
+ args.offset,
+ std::num::NonZeroU64::new(args.size),
+ );
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassSetVertexBufferArgs {
+ render_pass_rid: u32,
+ slot: u32,
+ buffer: u32,
+ offset: u64,
+ size: u64,
+}
+
+pub fn op_webgpu_render_pass_set_vertex_buffer(
+ state: &mut OpState,
+ args: RenderPassSetVertexBufferArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let buffer_resource = state
+ .resource_table
+ .get::<super::buffer::WebGPUBuffer>(args.buffer)
+ .ok_or_else(bad_resource_id)?;
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ wgpu_core::command::render_ffi::wgpu_render_pass_set_vertex_buffer(
+ &mut render_pass_resource.0.borrow_mut(),
+ args.slot,
+ buffer_resource.0,
+ args.offset,
+ std::num::NonZeroU64::new(args.size),
+ );
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassDrawArgs {
+ render_pass_rid: u32,
+ vertex_count: u32,
+ instance_count: u32,
+ first_vertex: u32,
+ first_instance: u32,
+}
+
+pub fn op_webgpu_render_pass_draw(
+ state: &mut OpState,
+ args: RenderPassDrawArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ wgpu_core::command::render_ffi::wgpu_render_pass_draw(
+ &mut render_pass_resource.0.borrow_mut(),
+ args.vertex_count,
+ args.instance_count,
+ args.first_vertex,
+ args.first_instance,
+ );
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassDrawIndexedArgs {
+ render_pass_rid: u32,
+ index_count: u32,
+ instance_count: u32,
+ first_index: u32,
+ base_vertex: i32,
+ first_instance: u32,
+}
+
+pub fn op_webgpu_render_pass_draw_indexed(
+ state: &mut OpState,
+ args: RenderPassDrawIndexedArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ wgpu_core::command::render_ffi::wgpu_render_pass_draw_indexed(
+ &mut render_pass_resource.0.borrow_mut(),
+ args.index_count,
+ args.instance_count,
+ args.first_index,
+ args.base_vertex,
+ args.first_instance,
+ );
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassDrawIndirectArgs {
+ render_pass_rid: u32,
+ indirect_buffer: u32,
+ indirect_offset: u64,
+}
+
+pub fn op_webgpu_render_pass_draw_indirect(
+ state: &mut OpState,
+ args: RenderPassDrawIndirectArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let buffer_resource = state
+ .resource_table
+ .get::<super::buffer::WebGPUBuffer>(args.indirect_buffer)
+ .ok_or_else(bad_resource_id)?;
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ wgpu_core::command::render_ffi::wgpu_render_pass_draw_indirect(
+ &mut render_pass_resource.0.borrow_mut(),
+ buffer_resource.0,
+ args.indirect_offset,
+ );
+
+ Ok(json!({}))
+}
+
+#[derive(Deserialize)]
+#[serde(rename_all = "camelCase")]
+pub struct RenderPassDrawIndexedIndirectArgs {
+ render_pass_rid: u32,
+ indirect_buffer: u32,
+ indirect_offset: u64,
+}
+
+pub fn op_webgpu_render_pass_draw_indexed_indirect(
+ state: &mut OpState,
+ args: RenderPassDrawIndexedIndirectArgs,
+ _zero_copy: &mut [ZeroCopyBuf],
+) -> Result<Value, AnyError> {
+ let buffer_resource = state
+ .resource_table
+ .get::<super::buffer::WebGPUBuffer>(args.indirect_buffer)
+ .ok_or_else(bad_resource_id)?;
+ let render_pass_resource = state
+ .resource_table
+ .get::<WebGPURenderPass>(args.render_pass_rid)
+ .ok_or_else(bad_resource_id)?;
+
+ wgpu_core::command::render_ffi::wgpu_render_pass_draw_indexed_indirect(
+ &mut render_pass_resource.0.borrow_mut(),
+ buffer_resource.0,
+ args.indirect_offset,
+ );
+
+ Ok(json!({}))
+}