Files
wgpu/examples/convert.rs
João Capucho 0aae8ae6f8 glsl-out: More glsl versions (#191)
* Added support for glsl core version up to 400

* Added 330 to the list supported versions

* Added links to all extensions and fixed a extension being wrongfully activated

* Cleanup and check for image formats

* Removed useless extension and location on layout

* Added support for external functions and fixed global names

* Typefier allow Scalars in distance and length

* Addressed comments

* Fixed extension appearing after body

* Fixed entry point not being generated
2020-09-18 17:01:05 -04:00

205 lines
6.3 KiB
Rust

use serde::{Deserialize, Serialize};
use std::{env, fs, path::Path};
#[derive(Hash, PartialEq, Eq, Serialize, Deserialize)]
struct BindSource {
group: u32,
binding: u32,
}
#[derive(Serialize, Deserialize)]
struct BindTarget {
#[serde(default)]
buffer: Option<u8>,
#[serde(default)]
texture: Option<u8>,
#[serde(default)]
sampler: Option<u8>,
#[serde(default)]
mutable: bool,
}
#[derive(Default, Serialize, Deserialize)]
struct Parameters {
metal_bindings: naga::FastHashMap<BindSource, BindTarget>,
}
fn main() {
env_logger::init();
let args = env::args().collect::<Vec<_>>();
if args.len() <= 1 {
println!("Call with <input> <output>");
return;
}
let module = match Path::new(&args[1])
.extension()
.expect("Input has no extension?")
.to_str()
.unwrap()
{
#[cfg(feature = "spv-in")]
"spv" => {
let input = fs::read(&args[1]).unwrap();
naga::front::spv::parse_u8_slice(&input).unwrap()
}
#[cfg(feature = "wgsl-in")]
"wgsl" => {
let input = fs::read_to_string(&args[1]).unwrap();
naga::front::wgsl::parse_str(&input).unwrap()
}
#[cfg(feature = "glsl-in")]
"vert" => {
let input = fs::read_to_string(&args[1]).unwrap();
naga::front::glsl::parse_str(&input, "main", naga::ShaderStage::Vertex).unwrap()
}
#[cfg(feature = "glsl-in")]
"frag" => {
let input = fs::read_to_string(&args[1]).unwrap();
naga::front::glsl::parse_str(&input, "main", naga::ShaderStage::Fragment).unwrap()
}
#[cfg(feature = "glsl-in")]
"comp" => {
let input = fs::read_to_string(&args[1]).unwrap();
naga::front::glsl::parse_str(&input, "main", naga::ShaderStage::Compute).unwrap()
}
#[cfg(feature = "deserialize")]
"ron" => {
let mut input = fs::File::open(&args[1]).unwrap();
ron::de::from_reader(&mut input).unwrap()
}
other => {
if true {
// prevent "unreachable_code" warnings
panic!("Unknown input extension: {}", other);
}
naga::Module::generate_empty()
}
};
if args.len() <= 2 {
println!("{:#?}", module);
return;
}
let param_path = std::path::PathBuf::from(&args[1]).with_extension("param.ron");
let params = match fs::read_to_string(param_path) {
Ok(string) => ron::de::from_str(&string).unwrap(),
Err(_) => Parameters::default(),
};
match Path::new(&args[2])
.extension()
.expect("Output has no extension?")
.to_str()
.unwrap()
{
#[cfg(feature = "msl-out")]
"metal" => {
use naga::back::msl;
let mut binding_map = msl::BindingMap::default();
for (key, value) in params.metal_bindings {
binding_map.insert(
msl::BindSource {
group: key.group,
binding: key.binding,
},
msl::BindTarget {
buffer: value.buffer,
texture: value.texture,
sampler: value.sampler,
mutable: value.mutable,
},
);
}
let options = msl::Options {
binding_map: &binding_map,
};
let msl = msl::write_string(&module, options).unwrap();
fs::write(&args[2], msl).unwrap();
}
#[cfg(feature = "spv-out")]
"spv" => {
use naga::back::spv;
let debug_flag = args.get(3).map_or(spv::WriterFlags::DEBUG, |arg| {
if arg.parse().unwrap() {
spv::WriterFlags::DEBUG
} else {
spv::WriterFlags::NONE
}
});
let spv = spv::Writer::new(&module.header, debug_flag).write(&module);
let bytes = spv
.iter()
.fold(Vec::with_capacity(spv.len() * 4), |mut v, w| {
v.extend_from_slice(&w.to_le_bytes());
v
});
fs::write(&args[2], bytes.as_slice()).unwrap();
}
#[cfg(feature = "glsl-out")]
stage @ "vert" | stage @ "frag" | stage @ "comp" => {
use naga::{
back::glsl::{self, Options, Version},
ShaderStage,
};
let version = match args.get(3).map(|p| p.as_str()) {
Some("core") => {
Version::Desktop(args.get(4).and_then(|v| v.parse().ok()).unwrap_or(330))
}
Some("es") => {
Version::Embedded(args.get(4).and_then(|v| v.parse().ok()).unwrap_or(310))
}
Some(_) => panic!("Unknown profile"),
_ => Version::Embedded(310),
};
let options = Options {
version,
entry_point: (
match stage {
"vert" => ShaderStage::Vertex,
"frag" => ShaderStage::Fragment,
"comp" => ShaderStage::Compute,
_ => unreachable!(),
},
String::from("main"),
),
};
let mut file = fs::OpenOptions::new()
.write(true)
.truncate(true)
.create(true)
.open(&args[2])
.unwrap();
glsl::write(&module, &mut file, options)
.map_err(|e| {
fs::remove_file(&args[2]).unwrap();
e
})
.unwrap();
}
#[cfg(feature = "serialize")]
"ron" => {
let config = ron::ser::PrettyConfig::new()
.with_enumerate_arrays(true)
.with_decimal_floats(true);
let output = ron::ser::to_string_pretty(&module, config).unwrap();
fs::write(&args[2], output).unwrap();
}
other => {
let _ = params;
panic!("Unknown output extension: {}", other);
}
}
}