2021-03-01 11:31:13 +01:00
|
|
|
// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.
|
|
|
|
|
|
|
|
use deno_core::error::bad_resource_id;
|
2021-04-02 15:47:57 +02:00
|
|
|
use deno_core::error::null_opbuf;
|
2021-03-01 11:31:13 +01:00
|
|
|
use deno_core::error::AnyError;
|
2021-03-19 22:55:37 +05:30
|
|
|
use deno_core::ResourceId;
|
2021-03-01 11:31:13 +01:00
|
|
|
use deno_core::ZeroCopyBuf;
|
|
|
|
use deno_core::{OpState, Resource};
|
|
|
|
use serde::Deserialize;
|
|
|
|
use std::borrow::Cow;
|
|
|
|
|
2021-04-05 18:40:24 +02:00
|
|
|
use super::error::WebGpuResult;
|
2021-03-01 11:31:13 +01:00
|
|
|
|
2021-03-26 03:17:37 +09:00
|
|
|
pub(crate) struct WebGpuShaderModule(pub(crate) wgpu_core::id::ShaderModuleId);
|
|
|
|
impl Resource for WebGpuShaderModule {
|
2021-03-01 11:31:13 +01:00
|
|
|
fn name(&self) -> Cow<str> {
|
|
|
|
"webGPUShaderModule".into()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Deserialize)]
|
|
|
|
#[serde(rename_all = "camelCase")]
|
|
|
|
pub struct CreateShaderModuleArgs {
|
2021-03-19 22:55:37 +05:30
|
|
|
device_rid: ResourceId,
|
2021-03-01 11:31:13 +01:00
|
|
|
label: Option<String>,
|
|
|
|
code: Option<String>,
|
|
|
|
_source_map: Option<()>, // not yet implemented
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn op_webgpu_create_shader_module(
|
|
|
|
state: &mut OpState,
|
|
|
|
args: CreateShaderModuleArgs,
|
2021-04-02 15:47:57 +02:00
|
|
|
zero_copy: Option<ZeroCopyBuf>,
|
2021-04-05 18:40:24 +02:00
|
|
|
) -> Result<WebGpuResult, AnyError> {
|
2021-03-01 11:31:13 +01:00
|
|
|
let instance = state.borrow::<super::Instance>();
|
|
|
|
let device_resource = state
|
|
|
|
.resource_table
|
2021-03-26 03:17:37 +09:00
|
|
|
.get::<super::WebGpuDevice>(args.device_rid)
|
2021-03-01 11:31:13 +01:00
|
|
|
.ok_or_else(bad_resource_id)?;
|
|
|
|
let device = device_resource.0;
|
|
|
|
|
|
|
|
let source = match args.code {
|
|
|
|
Some(code) => {
|
|
|
|
wgpu_core::pipeline::ShaderModuleSource::Wgsl(Cow::from(code))
|
|
|
|
}
|
|
|
|
None => wgpu_core::pipeline::ShaderModuleSource::SpirV(Cow::from(unsafe {
|
2021-04-02 15:47:57 +02:00
|
|
|
match &zero_copy {
|
|
|
|
Some(zero_copy) => {
|
|
|
|
let (prefix, data, suffix) = zero_copy.align_to::<u32>();
|
|
|
|
assert!(prefix.is_empty());
|
|
|
|
assert!(suffix.is_empty());
|
|
|
|
data
|
|
|
|
}
|
|
|
|
None => return Err(null_opbuf()),
|
|
|
|
}
|
2021-03-01 11:31:13 +01:00
|
|
|
})),
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut flags = wgpu_types::ShaderFlags::default();
|
|
|
|
flags.set(wgpu_types::ShaderFlags::VALIDATION, true);
|
|
|
|
#[cfg(all(target_os = "macos", target_arch = "x86_64"))]
|
|
|
|
flags.set(wgpu_types::ShaderFlags::EXPERIMENTAL_TRANSLATION, true);
|
|
|
|
|
|
|
|
let descriptor = wgpu_core::pipeline::ShaderModuleDescriptor {
|
|
|
|
label: args.label.map(Cow::from),
|
|
|
|
flags,
|
|
|
|
};
|
|
|
|
|
|
|
|
let (shader_module, maybe_err) = gfx_select!(device => instance.device_create_shader_module(
|
|
|
|
device,
|
|
|
|
&descriptor,
|
|
|
|
source,
|
|
|
|
std::marker::PhantomData
|
|
|
|
));
|
|
|
|
|
2021-03-26 03:17:37 +09:00
|
|
|
let rid = state.resource_table.add(WebGpuShaderModule(shader_module));
|
2021-03-01 11:31:13 +01:00
|
|
|
|
2021-04-05 18:40:24 +02:00
|
|
|
Ok(WebGpuResult::rid_err(rid, maybe_err))
|
2021-03-01 11:31:13 +01:00
|
|
|
}
|