wgpu/deno_webgpu/queue.rs

127 lines
3.5 KiB
Rust
Raw Normal View History

// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
use deno_core::error::AnyError;
use deno_core::op;
use deno_core::OpState;
use deno_core::ResourceId;
use deno_core::ZeroCopyBuf;
use serde::Deserialize;
use super::error::WebGpuResult;
type WebGpuQueue = super::WebGpuDevice;
#[op]
pub fn op_webgpu_queue_submit(
state: &mut OpState,
2022-03-24 17:20:10 +00:00
queue_rid: ResourceId,
command_buffers: Vec<ResourceId>,
) -> Result<WebGpuResult, AnyError> {
let instance = state.borrow::<super::Instance>();
2022-03-24 17:20:10 +00:00
let queue_resource = state.resource_table.get::<WebGpuQueue>(queue_rid)?;
let queue = queue_resource.1;
2022-03-24 17:20:10 +00:00
let ids = command_buffers
.iter()
.map(|rid| {
let buffer_resource = state
.resource_table
.get::<super::command_encoder::WebGpuCommandBuffer>(*rid)?;
let mut id = buffer_resource.1.borrow_mut();
Ok(id.take().unwrap())
2022-03-24 17:20:10 +00:00
})
.collect::<Result<Vec<_>, AnyError>>()?;
let maybe_err = gfx_select!(queue => instance.queue_submit(queue, &ids)).err();
2022-03-24 17:20:10 +00:00
for rid in command_buffers {
state.resource_table.close(rid)?;
}
Ok(WebGpuResult::maybe_err(maybe_err))
}
#[derive(Deserialize)]
#[serde(rename_all = "camelCase")]
2022-03-24 17:20:10 +00:00
pub struct GpuImageDataLayout {
offset: u64,
bytes_per_row: Option<u32>,
rows_per_image: Option<u32>,
}
impl From<GpuImageDataLayout> for wgpu_types::ImageDataLayout {
fn from(layout: GpuImageDataLayout) -> Self {
wgpu_types::ImageDataLayout {
offset: layout.offset,
bytes_per_row: layout.bytes_per_row,
rows_per_image: layout.rows_per_image,
}
}
}
2022-03-24 17:20:10 +00:00
#[op]
pub fn op_webgpu_write_buffer(
state: &mut OpState,
queue_rid: ResourceId,
buffer: ResourceId,
buffer_offset: u64,
data_offset: usize,
size: Option<usize>,
2022-03-24 17:20:10 +00:00
buf: ZeroCopyBuf,
) -> Result<WebGpuResult, AnyError> {
let instance = state.borrow::<super::Instance>();
let buffer_resource = state
.resource_table
2022-03-24 17:20:10 +00:00
.get::<super::buffer::WebGpuBuffer>(buffer)?;
let buffer = buffer_resource.1;
2022-03-24 17:20:10 +00:00
let queue_resource = state.resource_table.get::<WebGpuQueue>(queue_rid)?;
let queue = queue_resource.1;
2022-03-24 17:20:10 +00:00
let data = match size {
Some(size) => &buf[data_offset..(data_offset + size)],
None => &buf[data_offset..],
};
let maybe_err = gfx_select!(queue => instance.queue_write_buffer(
queue,
buffer,
2022-03-24 17:20:10 +00:00
buffer_offset,
data
))
.err();
Ok(WebGpuResult::maybe_err(maybe_err))
}
2022-03-24 17:20:10 +00:00
#[op]
pub fn op_webgpu_write_texture(
state: &mut OpState,
queue_rid: ResourceId,
destination: super::command_encoder::GpuImageCopyTexture,
data_layout: GpuImageDataLayout,
size: wgpu_types::Extent3d,
2022-03-24 17:20:10 +00:00
buf: ZeroCopyBuf,
) -> Result<WebGpuResult, AnyError> {
let instance = state.borrow::<super::Instance>();
let texture_resource = state
.resource_table
2022-03-24 17:20:10 +00:00
.get::<super::texture::WebGpuTexture>(destination.texture)?;
let queue_resource = state.resource_table.get::<WebGpuQueue>(queue_rid)?;
let queue = queue_resource.1;
let destination = wgpu_core::command::ImageCopyTexture {
texture: texture_resource.id,
2022-03-24 17:20:10 +00:00
mip_level: destination.mip_level,
origin: destination.origin,
aspect: destination.aspect,
};
2022-03-24 17:20:10 +00:00
let data_layout = data_layout.into();
gfx_ok!(queue => instance.queue_write_texture(
queue,
&destination,
2022-03-24 17:20:10 +00:00
&*buf,
&data_layout,
2022-03-24 17:20:10 +00:00
&size
))
}