#![feature(proc_macro_span)]
#![feature(proc_macro_diagnostic)]
use proc_macro::{TokenStream, Span, Diagnostic, Level};
use syn::{parse_macro_input, LitStr};
use naga::{
valid::{ValidationFlags, Validator, Capabilities},
front::wgsl,
};
#[cfg(feature = "spv-out")]
use::naga::back::spv::{self, Options, WriterFlags};
#[proc_macro]
pub fn include_wgsl(input: TokenStream) -> TokenStream {
let file_path = parse_macro_input!(input as LitStr).value();
let call_site = Span::call_site();
let mut own_path = call_site.source_file().path();
assert!(own_path.pop());
let new_path = own_path.join(&file_path);
match std::fs::read_to_string(new_path) {
Ok(wgsl_str) => {
match wgsl::parse_str(&wgsl_str) {
Ok(module) => {
match Validator::new(ValidationFlags::all(), Capabilities::all()).validate(&module) {
Ok(_) => {},
Err(e) => Diagnostic::new(Level::Error, format!("{}: {}", file_path, e)).emit(),
}
},
Err(e) => {
Diagnostic::new(Level::Error, format!("Unable to parse {}:", file_path)).emit();
e.emit_to_stderr(&wgsl_str);
},
}
},
Err(e) => {
Diagnostic::spanned(call_site, Level::Error, format!("couldn't read {}: {}", file_path, e));
},
};
format!("&include_str!(\"{}\")", file_path).parse().unwrap()
}
#[cfg(feature = "spv-out")]
#[proc_macro]
pub fn include_wgsl_to_spv(input: TokenStream) -> TokenStream {
let file_path = parse_macro_input!(input as LitStr).value();
let call_site = Span::call_site();
let mut own_path = call_site.source_file().path();
assert!(own_path.pop());
let new_path = own_path.join(&file_path);
match std::fs::read_to_string(new_path) {
Ok(wgsl_str) => {
match wgsl::parse_str(&wgsl_str) {
Ok(module) => {
match Validator::new(ValidationFlags::all(), Capabilities::all()).validate(&module) {
Ok(info) => {
#[cfg(not(debug_assertions))]
let flags = WriterFlags::LABEL_VARYINGS | WriterFlags::CLAMP_FRAG_DEPTH;
#[cfg(debug_assertions)]
let flags = WriterFlags::LABEL_VARYINGS | WriterFlags::CLAMP_FRAG_DEPTH | WriterFlags::DEBUG;
let spv_bytes = spv::write_vec(&module, &info, &Options {
flags,
..Default::default()
}, None).expect("Could not generate SPV array");
return format!("{{ include_str!(\"{}\"); {:?} }}", file_path, spv_bytes).parse().expect("Cannot format return SPV array");
},
Err(e) => {
Diagnostic::new(Level::Error, format!("{}: {}", file_path, e)).emit()
},
}
},
Err(e) => {
Diagnostic::new(Level::Error, format!("Unable to parse {}:", file_path)).emit();
e.emit_to_stderr(&wgsl_str);
},
}
},
Err(e) => {
Diagnostic::spanned(call_site, Level::Error, format!("couldn't read {}: {}", file_path, e));
},
};
format!("{{ include_str!(\"{}\"); [] }}", file_path).parse().unwrap()
}