mirror of
https://github.com/gfx-rs/wgpu.git
synced 2024-11-27 17:23:46 +00:00
608 lines
18 KiB
Rust
608 lines
18 KiB
Rust
//TODO: move this to a binary target once Rust supports
|
|
// binary-specific dependencies.
|
|
|
|
use std::{fs, path::Path, path::PathBuf};
|
|
|
|
const BASE_DIR_IN: &str = "tests/in";
|
|
const BASE_DIR_OUT: &str = "tests/out";
|
|
|
|
bitflags::bitflags! {
|
|
struct Targets: u32 {
|
|
const IR = 0x1;
|
|
const ANALYSIS = 0x2;
|
|
const SPIRV = 0x4;
|
|
const METAL = 0x8;
|
|
const GLSL = 0x10;
|
|
const DOT = 0x20;
|
|
const HLSL = 0x40;
|
|
const WGSL = 0x80;
|
|
}
|
|
}
|
|
|
|
/// Twin of `naga::back::BoundsCheckPolicy`, always serializable.
|
|
#[derive(Clone, Copy, serde::Deserialize)]
|
|
enum BoundsCheckPolicyArg {
|
|
Restrict,
|
|
ReadZeroSkipWrite,
|
|
Unchecked,
|
|
}
|
|
|
|
impl Default for BoundsCheckPolicyArg {
|
|
fn default() -> Self {
|
|
BoundsCheckPolicyArg::Unchecked
|
|
}
|
|
}
|
|
|
|
impl From<BoundsCheckPolicyArg> for naga::back::BoundsCheckPolicy {
|
|
fn from(arg: BoundsCheckPolicyArg) -> Self {
|
|
match arg {
|
|
BoundsCheckPolicyArg::Restrict => Self::Restrict,
|
|
BoundsCheckPolicyArg::ReadZeroSkipWrite => Self::ReadZeroSkipWrite,
|
|
BoundsCheckPolicyArg::Unchecked => Self::Unchecked,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Default, serde::Deserialize)]
|
|
struct Parameters {
|
|
#[serde(default)]
|
|
god_mode: bool,
|
|
|
|
#[allow(dead_code)]
|
|
#[serde(default)]
|
|
index_bounds_check_policy: BoundsCheckPolicyArg,
|
|
#[allow(dead_code)]
|
|
#[serde(default)]
|
|
buffer_bounds_check_policy: BoundsCheckPolicyArg,
|
|
#[allow(dead_code)]
|
|
#[serde(default)]
|
|
image_bounds_check_policy: BoundsCheckPolicyArg,
|
|
|
|
#[cfg_attr(not(feature = "spv-out"), allow(dead_code))]
|
|
spv_version: (u8, u8),
|
|
#[cfg_attr(not(feature = "spv-out"), allow(dead_code))]
|
|
#[serde(default)]
|
|
spv_capabilities: naga::FastHashSet<spirv::Capability>,
|
|
#[cfg_attr(not(feature = "spv-out"), allow(dead_code))]
|
|
#[serde(default)]
|
|
spv_debug: bool,
|
|
#[cfg_attr(not(feature = "spv-out"), allow(dead_code))]
|
|
#[serde(default)]
|
|
spv_adjust_coordinate_space: bool,
|
|
#[cfg_attr(not(feature = "spv-out"), allow(dead_code))]
|
|
#[serde(default)]
|
|
spv_separate_entry_points: bool,
|
|
#[cfg(all(feature = "deserialize", feature = "msl-out"))]
|
|
#[serde(default)]
|
|
msl: naga::back::msl::Options,
|
|
#[cfg(all(not(feature = "deserialize"), feature = "msl-out"))]
|
|
#[serde(default)]
|
|
msl_custom: bool,
|
|
#[cfg(all(feature = "deserialize", feature = "glsl-out"))]
|
|
#[serde(default)]
|
|
glsl: naga::back::glsl::Options,
|
|
#[cfg(all(not(feature = "deserialize"), feature = "glsl-out"))]
|
|
#[serde(default)]
|
|
glsl_custom: bool,
|
|
#[cfg(all(feature = "deserialize", feature = "hlsl-out"))]
|
|
#[serde(default)]
|
|
hlsl: naga::back::hlsl::Options,
|
|
#[cfg(all(not(feature = "deserialize"), feature = "hlsl-out"))]
|
|
#[serde(default)]
|
|
hlsl_custom: bool,
|
|
}
|
|
|
|
#[allow(dead_code, unused_variables)]
|
|
fn check_targets(module: &naga::Module, name: &str, targets: Targets) {
|
|
let root = env!("CARGO_MANIFEST_DIR");
|
|
let params = match fs::read_to_string(format!("{}/{}/{}.param.ron", root, BASE_DIR_IN, name)) {
|
|
Ok(string) => ron::de::from_str(&string).expect("Couldn't parse param file"),
|
|
Err(_) => Parameters::default(),
|
|
};
|
|
let capabilities = if params.god_mode {
|
|
naga::valid::Capabilities::all()
|
|
} else {
|
|
naga::valid::Capabilities::empty()
|
|
};
|
|
let info = naga::valid::Validator::new(naga::valid::ValidationFlags::all(), capabilities)
|
|
.validate(module)
|
|
.unwrap();
|
|
|
|
let dest = PathBuf::from(root).join(BASE_DIR_OUT);
|
|
|
|
#[cfg(feature = "serialize")]
|
|
{
|
|
if targets.contains(Targets::IR) {
|
|
let config = ron::ser::PrettyConfig::default().with_new_line("\n".to_string());
|
|
let string = ron::ser::to_string_pretty(module, config).unwrap();
|
|
fs::write(dest.join(format!("ir/{}.ron", name)), string).unwrap();
|
|
}
|
|
if targets.contains(Targets::ANALYSIS) {
|
|
let config = ron::ser::PrettyConfig::default().with_new_line("\n".to_string());
|
|
let string = ron::ser::to_string_pretty(&info, config).unwrap();
|
|
fs::write(dest.join(format!("analysis/{}.info.ron", name)), string).unwrap();
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "spv-out")]
|
|
{
|
|
if targets.contains(Targets::SPIRV) {
|
|
write_output_spv(module, &info, &dest, name, ¶ms);
|
|
}
|
|
}
|
|
#[cfg(feature = "msl-out")]
|
|
{
|
|
if targets.contains(Targets::METAL) {
|
|
write_output_msl(module, &info, &dest, name, ¶ms);
|
|
}
|
|
}
|
|
#[cfg(feature = "glsl-out")]
|
|
{
|
|
if targets.contains(Targets::GLSL) {
|
|
for ep in module.entry_points.iter() {
|
|
write_output_glsl(module, &info, &dest, name, ep.stage, &ep.name, ¶ms);
|
|
}
|
|
}
|
|
}
|
|
#[cfg(feature = "dot-out")]
|
|
{
|
|
if targets.contains(Targets::DOT) {
|
|
let string = naga::back::dot::write(module, Some(&info)).unwrap();
|
|
fs::write(dest.join(format!("dot/{}.dot", name)), string).unwrap();
|
|
}
|
|
}
|
|
#[cfg(feature = "hlsl-out")]
|
|
{
|
|
if targets.contains(Targets::HLSL) {
|
|
write_output_hlsl(module, &info, &dest, name, ¶ms);
|
|
}
|
|
}
|
|
#[cfg(feature = "wgsl-out")]
|
|
{
|
|
if targets.contains(Targets::WGSL) {
|
|
write_output_wgsl(module, &info, &dest, name);
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "spv-out")]
|
|
fn write_output_spv(
|
|
module: &naga::Module,
|
|
info: &naga::valid::ModuleInfo,
|
|
destination: &Path,
|
|
file_name: &str,
|
|
params: &Parameters,
|
|
) {
|
|
use naga::back::spv;
|
|
use rspirv::binary::Disassemble;
|
|
|
|
let mut flags = spv::WriterFlags::empty();
|
|
if params.spv_debug {
|
|
flags |= spv::WriterFlags::DEBUG;
|
|
}
|
|
if params.spv_adjust_coordinate_space {
|
|
flags |= spv::WriterFlags::ADJUST_COORDINATE_SPACE;
|
|
}
|
|
let options = spv::Options {
|
|
lang_version: params.spv_version,
|
|
flags,
|
|
capabilities: if params.spv_capabilities.is_empty() {
|
|
None
|
|
} else {
|
|
Some(params.spv_capabilities.clone())
|
|
},
|
|
|
|
bounds_check_policies: naga::back::BoundsCheckPolicies {
|
|
index: params.index_bounds_check_policy.into(),
|
|
buffer: params.buffer_bounds_check_policy.into(),
|
|
image: params.image_bounds_check_policy.into(),
|
|
},
|
|
|
|
..spv::Options::default()
|
|
};
|
|
|
|
if params.spv_separate_entry_points {
|
|
for ep in module.entry_points.iter() {
|
|
let pipeline_options = spv::PipelineOptions {
|
|
entry_point: ep.name.clone(),
|
|
shader_stage: ep.stage,
|
|
};
|
|
let spv = spv::write_vec(module, info, &options, Some(&pipeline_options)).unwrap();
|
|
let dis = rspirv::dr::load_words(spv)
|
|
.expect("Produced invalid SPIR-V")
|
|
.disassemble();
|
|
let path = format!("spv/{}.{}.spvasm", file_name, ep.name);
|
|
fs::write(destination.join(path), dis).unwrap();
|
|
}
|
|
} else {
|
|
let spv = spv::write_vec(module, info, &options, None).unwrap();
|
|
let dis = rspirv::dr::load_words(spv)
|
|
.expect("Produced invalid SPIR-V")
|
|
.disassemble();
|
|
fs::write(destination.join(format!("spv/{}.spvasm", file_name)), dis).unwrap();
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "msl-out")]
|
|
fn write_output_msl(
|
|
module: &naga::Module,
|
|
info: &naga::valid::ModuleInfo,
|
|
destination: &Path,
|
|
file_name: &str,
|
|
params: &Parameters,
|
|
) {
|
|
use naga::back::msl;
|
|
|
|
#[cfg_attr(feature = "deserialize", allow(unused_variables))]
|
|
let default_options = msl::Options::default();
|
|
#[cfg(feature = "deserialize")]
|
|
let options = ¶ms.msl;
|
|
#[cfg(not(feature = "deserialize"))]
|
|
let options = if params.msl_custom {
|
|
println!("Skipping {}", destination.display());
|
|
return;
|
|
} else {
|
|
&default_options
|
|
};
|
|
|
|
let pipeline_options = msl::PipelineOptions {
|
|
allow_point_size: true,
|
|
};
|
|
|
|
let (string, tr_info) = msl::write_string(module, info, options, &pipeline_options).unwrap();
|
|
|
|
for (ep, result) in module.entry_points.iter().zip(tr_info.entry_point_names) {
|
|
if let Err(error) = result {
|
|
panic!("Failed to translate '{}': {}", ep.name, error);
|
|
}
|
|
}
|
|
|
|
fs::write(destination.join(format!("msl/{}.msl", file_name)), string).unwrap();
|
|
}
|
|
|
|
#[cfg(feature = "glsl-out")]
|
|
fn write_output_glsl(
|
|
module: &naga::Module,
|
|
info: &naga::valid::ModuleInfo,
|
|
destination: &Path,
|
|
file_name: &str,
|
|
stage: naga::ShaderStage,
|
|
ep_name: &str,
|
|
params: &Parameters,
|
|
) {
|
|
use naga::back::glsl;
|
|
|
|
#[cfg_attr(feature = "deserialize", allow(unused_variables))]
|
|
let default_options = glsl::Options::default();
|
|
#[cfg(feature = "deserialize")]
|
|
let options = ¶ms.glsl;
|
|
#[cfg(not(feature = "deserialize"))]
|
|
let options = if params.glsl_custom {
|
|
println!("Skipping {}", destination.display());
|
|
return;
|
|
} else {
|
|
&default_options
|
|
};
|
|
|
|
let pipeline_options = glsl::PipelineOptions {
|
|
shader_stage: stage,
|
|
entry_point: ep_name.to_string(),
|
|
};
|
|
|
|
let mut buffer = String::new();
|
|
let mut writer =
|
|
glsl::Writer::new(&mut buffer, module, info, options, &pipeline_options).unwrap();
|
|
writer.write().unwrap();
|
|
|
|
fs::write(
|
|
destination.join(format!("glsl/{}.{}.{:?}.glsl", file_name, ep_name, stage)),
|
|
buffer,
|
|
)
|
|
.unwrap();
|
|
}
|
|
|
|
#[cfg(feature = "hlsl-out")]
|
|
fn write_output_hlsl(
|
|
module: &naga::Module,
|
|
info: &naga::valid::ModuleInfo,
|
|
destination: &Path,
|
|
file_name: &str,
|
|
params: &Parameters,
|
|
) {
|
|
use naga::back::hlsl;
|
|
use std::fmt::Write;
|
|
|
|
#[cfg_attr(feature = "deserialize", allow(unused_variables))]
|
|
let default_options = hlsl::Options::default();
|
|
#[cfg(feature = "deserialize")]
|
|
let options = ¶ms.hlsl;
|
|
#[cfg(not(feature = "deserialize"))]
|
|
let options = if params.hlsl_custom {
|
|
println!("Skipping {}", destination.display());
|
|
return;
|
|
} else {
|
|
&default_options
|
|
};
|
|
|
|
let mut buffer = String::new();
|
|
let mut writer = hlsl::Writer::new(&mut buffer, options);
|
|
let reflection_info = writer.write(module, info).unwrap();
|
|
|
|
fs::write(destination.join(format!("hlsl/{}.hlsl", file_name)), buffer).unwrap();
|
|
|
|
// We need a config file for validation script
|
|
// This file contains an info about profiles (shader stages) contains inside generated shader
|
|
// This info will be passed to dxc
|
|
let mut config_str = String::new();
|
|
let mut vertex_str = String::from("vertex=(");
|
|
let mut fragment_str = String::from("fragment=(");
|
|
let mut compute_str = String::from("compute=(");
|
|
for (index, ep) in module.entry_points.iter().enumerate() {
|
|
let name = match reflection_info.entry_point_names[index] {
|
|
Ok(ref name) => name,
|
|
Err(_) => continue,
|
|
};
|
|
match ep.stage {
|
|
naga::ShaderStage::Vertex => {
|
|
write!(
|
|
vertex_str,
|
|
"{}:{}_{} ",
|
|
name,
|
|
ep.stage.to_hlsl_str(),
|
|
options.shader_model.to_str(),
|
|
)
|
|
.unwrap();
|
|
}
|
|
naga::ShaderStage::Fragment => {
|
|
write!(
|
|
fragment_str,
|
|
"{}:{}_{} ",
|
|
name,
|
|
ep.stage.to_hlsl_str(),
|
|
options.shader_model.to_str(),
|
|
)
|
|
.unwrap();
|
|
}
|
|
naga::ShaderStage::Compute => {
|
|
write!(
|
|
compute_str,
|
|
"{}:{}_{} ",
|
|
name,
|
|
ep.stage.to_hlsl_str(),
|
|
options.shader_model.to_str(),
|
|
)
|
|
.unwrap();
|
|
}
|
|
}
|
|
}
|
|
|
|
writeln!(
|
|
config_str,
|
|
"{})\n{})\n{})",
|
|
vertex_str, fragment_str, compute_str
|
|
)
|
|
.unwrap();
|
|
|
|
fs::write(
|
|
destination.join(format!("hlsl/{}.hlsl.config", file_name)),
|
|
config_str,
|
|
)
|
|
.unwrap();
|
|
}
|
|
|
|
#[cfg(feature = "wgsl-out")]
|
|
fn write_output_wgsl(
|
|
module: &naga::Module,
|
|
info: &naga::valid::ModuleInfo,
|
|
destination: &Path,
|
|
file_name: &str,
|
|
) {
|
|
use naga::back::wgsl;
|
|
|
|
let string = wgsl::write_string(module, info).unwrap();
|
|
|
|
fs::write(destination.join(format!("wgsl/{}.wgsl", file_name)), string).unwrap();
|
|
}
|
|
|
|
#[cfg(feature = "wgsl-in")]
|
|
#[test]
|
|
fn convert_wgsl() {
|
|
let _ = env_logger::try_init();
|
|
|
|
let root = env!("CARGO_MANIFEST_DIR");
|
|
let inputs = [
|
|
(
|
|
"empty",
|
|
Targets::SPIRV | Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
(
|
|
"quad",
|
|
Targets::SPIRV
|
|
| Targets::METAL
|
|
| Targets::GLSL
|
|
| Targets::DOT
|
|
| Targets::HLSL
|
|
| Targets::WGSL,
|
|
),
|
|
(
|
|
"boids",
|
|
Targets::SPIRV | Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
(
|
|
"skybox",
|
|
Targets::SPIRV | Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
(
|
|
"collatz",
|
|
Targets::SPIRV
|
|
| Targets::METAL
|
|
| Targets::IR
|
|
| Targets::ANALYSIS
|
|
| Targets::HLSL
|
|
| Targets::WGSL,
|
|
),
|
|
(
|
|
"shadow",
|
|
Targets::SPIRV | Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
(
|
|
"image",
|
|
Targets::SPIRV | Targets::METAL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
("extra", Targets::SPIRV | Targets::METAL | Targets::WGSL),
|
|
(
|
|
"operators",
|
|
Targets::SPIRV | Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
(
|
|
"interpolate",
|
|
Targets::SPIRV | Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
(
|
|
"access",
|
|
Targets::SPIRV | Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
(
|
|
"control-flow",
|
|
Targets::SPIRV | Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
(
|
|
"standard",
|
|
Targets::SPIRV | Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
//TODO: GLSL https://github.com/gfx-rs/naga/issues/874
|
|
(
|
|
"interface",
|
|
Targets::SPIRV | Targets::METAL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
(
|
|
"globals",
|
|
Targets::SPIRV | Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
("bounds-check-zero", Targets::SPIRV),
|
|
("bounds-check-image-restrict", Targets::SPIRV),
|
|
("bounds-check-image-rzsw", Targets::SPIRV),
|
|
("policy-mix", Targets::SPIRV),
|
|
(
|
|
"texture-arg",
|
|
Targets::SPIRV | Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
),
|
|
];
|
|
|
|
for &(name, targets) in inputs.iter() {
|
|
println!("Processing '{}'", name);
|
|
// WGSL shaders lives in root dir as a privileged.
|
|
let file = fs::read_to_string(format!("{}/{}/{}.wgsl", root, BASE_DIR_IN, name))
|
|
.expect("Couldn't find wgsl file");
|
|
match naga::front::wgsl::parse_str(&file) {
|
|
Ok(module) => check_targets(&module, name, targets),
|
|
Err(e) => panic!("{}", e.emit_to_string(&file)),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "spv-in")]
|
|
fn convert_spv(name: &str, adjust_coordinate_space: bool, targets: Targets) {
|
|
let _ = env_logger::try_init();
|
|
|
|
let root = env!("CARGO_MANIFEST_DIR");
|
|
let module = naga::front::spv::parse_u8_slice(
|
|
&fs::read(format!("{}/{}/spv/{}.spv", root, BASE_DIR_IN, name))
|
|
.expect("Couldn't find spv file"),
|
|
&naga::front::spv::Options {
|
|
adjust_coordinate_space,
|
|
strict_capabilities: false,
|
|
flow_graph_dump_prefix: None,
|
|
},
|
|
)
|
|
.unwrap();
|
|
check_targets(&module, name, targets);
|
|
naga::valid::Validator::new(
|
|
naga::valid::ValidationFlags::all(),
|
|
naga::valid::Capabilities::empty(),
|
|
)
|
|
.validate(&module)
|
|
.unwrap();
|
|
}
|
|
|
|
#[cfg(feature = "spv-in")]
|
|
#[test]
|
|
fn convert_spv_quad_vert() {
|
|
convert_spv(
|
|
"quad-vert",
|
|
false,
|
|
Targets::METAL | Targets::GLSL | Targets::HLSL | Targets::WGSL,
|
|
);
|
|
}
|
|
|
|
#[cfg(feature = "spv-in")]
|
|
#[test]
|
|
fn convert_spv_shadow() {
|
|
convert_spv("shadow", true, Targets::IR | Targets::ANALYSIS);
|
|
}
|
|
|
|
#[cfg(feature = "spv-in")]
|
|
#[test]
|
|
fn convert_spv_inverse_hyperbolic_trig_functions() {
|
|
convert_spv(
|
|
"inv-hyperbolic-trig-functions",
|
|
true,
|
|
Targets::HLSL | Targets::WGSL,
|
|
);
|
|
}
|
|
|
|
#[cfg(all(feature = "spv-in", feature = "spv-out"))]
|
|
#[test]
|
|
fn convert_spv_pointer_access() {
|
|
convert_spv("pointer-access", true, Targets::SPIRV);
|
|
}
|
|
|
|
#[cfg(feature = "glsl-in")]
|
|
#[allow(unused_variables)]
|
|
#[test]
|
|
fn convert_glsl_folder() {
|
|
let _ = env_logger::try_init();
|
|
|
|
let root = env!("CARGO_MANIFEST_DIR");
|
|
|
|
for entry in std::fs::read_dir(format!("{}/{}/glsl", root, BASE_DIR_IN)).unwrap() {
|
|
let entry = entry.unwrap();
|
|
let file_name = entry.file_name().into_string().unwrap();
|
|
|
|
if file_name.ends_with(".ron") {
|
|
// No needed to validate ron files
|
|
continue;
|
|
}
|
|
println!("Processing {}", file_name);
|
|
|
|
let mut parser = naga::front::glsl::Parser::default();
|
|
let module = parser
|
|
.parse(
|
|
&naga::front::glsl::Options {
|
|
stage: match entry.path().extension().and_then(|s| s.to_str()).unwrap() {
|
|
"vert" => naga::ShaderStage::Vertex,
|
|
"frag" => naga::ShaderStage::Fragment,
|
|
"comp" => naga::ShaderStage::Compute,
|
|
ext => panic!("Unknown extension for glsl file {}", ext),
|
|
},
|
|
defines: Default::default(),
|
|
},
|
|
&fs::read_to_string(entry.path()).expect("Couldn't find glsl file"),
|
|
)
|
|
.unwrap();
|
|
|
|
let info = naga::valid::Validator::new(
|
|
naga::valid::ValidationFlags::all(),
|
|
naga::valid::Capabilities::all(),
|
|
)
|
|
.validate(&module)
|
|
.unwrap();
|
|
|
|
#[cfg(feature = "wgsl-out")]
|
|
{
|
|
let dest = PathBuf::from(root).join(BASE_DIR_OUT);
|
|
write_output_wgsl(&module, &info, &dest, &file_name.replace(".", "-"));
|
|
}
|
|
}
|
|
}
|