librashader/librashader-preprocess/src/lib.rs

129 lines
3.8 KiB
Rust
Raw Normal View History

2023-01-18 19:39:36 -05:00
//! Shader preprocessing for librashader.
//!
//! This crate contains facilities and types for resolving `#include` directives in `.slang`
//! into a single compilation unit. `#pragma` directives are also parsed and resolved as
//! [`ShaderParameter`](crate::ShaderParameter) structs.
//!
//! The resulting [`ShaderSource`](crate::ShaderSource) can then be passed into a
//! reflection target for reflection and compilation into the target shader format.
//!
//! Re-exported as [`librashader::preprocess`](https://docs.rs/librashader/latest/librashader/preprocess/index.html).
mod error;
mod include;
2022-10-21 21:04:00 -04:00
mod pragma;
mod stage;
2022-10-23 00:59:18 -04:00
use crate::include::read_source;
pub use error::*;
2022-11-29 01:57:04 -05:00
use librashader_common::ImageFormat;
use rustc_hash::FxHashMap;
2023-01-15 18:21:01 -05:00
use std::path::Path;
2022-11-30 22:50:57 -05:00
/// The source file for a single shader pass.
#[derive(Debug, Clone, PartialEq)]
pub struct ShaderSource {
2022-11-30 22:50:57 -05:00
/// The source contents for the vertex shader.
pub vertex: String,
2022-11-30 22:50:57 -05:00
/// The source contents for the fragment shader.
pub fragment: String,
2022-11-30 22:50:57 -05:00
/// The alias of the shader if available.
pub name: Option<String>,
2022-11-30 22:50:57 -05:00
/// The list of shader parameters found in the shader source.
pub parameters: FxHashMap<String, ShaderParameter>,
2022-11-30 22:50:57 -05:00
/// The image format the shader expects.
2022-11-29 01:57:04 -05:00
pub format: ImageFormat,
}
2022-11-30 22:50:57 -05:00
/// A user tweakable parameter for the shader as declared in source.
#[derive(Debug, Clone, PartialEq)]
pub struct ShaderParameter {
2022-11-30 22:50:57 -05:00
/// The name of the parameter.
pub id: String,
2022-11-30 22:50:57 -05:00
/// The description of the parameter.
pub description: String,
2022-11-30 22:50:57 -05:00
/// The initial value the parameter is set to.
pub initial: f32,
2022-11-30 22:50:57 -05:00
/// The minimum value that the parameter can be set to.
pub minimum: f32,
2022-11-30 22:50:57 -05:00
/// The maximum value that the parameter can be set to.
pub maximum: f32,
2022-11-30 22:50:57 -05:00
/// The step by which this parameter can be incremented or decremented.
pub step: f32,
}
impl ShaderSource {
2022-11-30 22:50:57 -05:00
/// Load the source file at the given path, resolving includes relative to the location of the
/// source file.
pub fn load(path: impl AsRef<Path>) -> Result<ShaderSource, PreprocessError> {
load_shader_source(path)
}
}
pub(crate) trait SourceOutput {
fn push_line(&mut self, str: &str);
fn mark_line(&mut self, line_no: usize, comment: &str) {
2022-10-23 00:59:18 -04:00
#[cfg(feature = "line_directives")]
self.push_line(&format!("#line {line_no} \"{comment}\""))
}
}
impl SourceOutput for String {
fn push_line(&mut self, str: &str) {
self.push_str(str);
self.push('\n');
}
}
pub(crate) fn load_shader_source(path: impl AsRef<Path>) -> Result<ShaderSource, PreprocessError> {
let source = read_source(path)?;
let meta = pragma::parse_pragma_meta(&source)?;
let text = stage::process_stages(&source)?;
2023-01-15 18:21:01 -05:00
let parameters = FxHashMap::from_iter(meta.parameters.into_iter().map(|p| (p.id.clone(), p)));
Ok(ShaderSource {
vertex: text.vertex,
fragment: text.fragment,
name: meta.name,
parameters,
format: meta.format,
})
}
#[cfg(test)]
mod test {
use crate::include::read_source;
use crate::{load_shader_source, pragma};
#[test]
pub fn load_file() {
2022-10-23 00:59:18 -04:00
let result = load_shader_source(
"../test/slang-shaders/blurs/shaders/royale/blur3x3-last-pass.slang",
)
.unwrap();
eprintln!("{:#}", result.vertex)
}
2022-10-21 23:37:47 -04:00
#[test]
pub fn preprocess_file() {
2022-10-21 21:04:00 -04:00
let result =
read_source("../test/slang-shaders/blurs/shaders/royale/blur3x3-last-pass.slang")
.unwrap();
eprintln!("{result}")
}
2022-10-21 23:37:47 -04:00
#[test]
pub fn get_param_pragmas() {
2022-10-23 00:59:18 -04:00
let result = read_source(
"../test/slang-shaders/crt/shaders/crt-maximus-royale/src/ntsc_pass1.slang",
)
.unwrap();
2022-10-21 23:37:47 -04:00
2022-10-23 00:59:18 -04:00
let params = pragma::parse_pragma_meta(result).unwrap();
2022-10-21 23:37:47 -04:00
eprintln!("{params:?}")
}
2022-10-21 21:04:00 -04:00
}