Source code

Revision control

Copy as Markdown

Other Tools

use alloc::{borrow::Cow, boxed::Box, string::String};
use core::{error::Error, fmt};
#[derive(Clone, Debug)]
pub struct ShaderError<E> {
/// The source code of the shader.
pub source: String,
pub label: Option<String>,
pub inner: Box<E>,
}
#[cfg(feature = "wgsl-in")]
impl fmt::Display for ShaderError<crate::front::wgsl::ParseError> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let label = self.label.as_deref().unwrap_or_default();
let string = self.inner.emit_to_string(&self.source);
write!(f, "\nShader '{label}' parsing {string}")
}
}
#[cfg(feature = "glsl-in")]
impl fmt::Display for ShaderError<crate::front::glsl::ParseErrors> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let label = self.label.as_deref().unwrap_or_default();
let string = self.inner.emit_to_string(&self.source);
write!(f, "\nShader '{label}' parsing {string}")
}
}
#[cfg(feature = "spv-in")]
impl fmt::Display for ShaderError<crate::front::spv::Error> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let label = self.label.as_deref().unwrap_or_default();
let string = self.inner.emit_to_string(&self.source);
write!(f, "\nShader '{label}' parsing {string}")
}
}
impl fmt::Display for ShaderError<crate::WithSpan<crate::valid::ValidationError>> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
use codespan_reporting::{files::SimpleFile, term};
let label = self.label.as_deref().unwrap_or_default();
let files = SimpleFile::new(label, replace_control_chars(&self.source));
let config = term::Config::default();
let writer = {
let mut writer = DiagnosticBuffer::new();
term::emit(
writer.inner_mut(),
&config,
&files,
&self.inner.diagnostic(),
)
.expect("cannot write error");
writer.into_string()
};
write!(f, "\nShader validation {writer}")
}
}
cfg_if::cfg_if! {
if #[cfg(feature = "termcolor")] {
type DiagnosticBufferInner = codespan_reporting::term::termcolor::NoColor<alloc::vec::Vec<u8>>;
pub(crate) use codespan_reporting::term::termcolor::WriteColor as _ErrorWrite;
} else if #[cfg(feature = "stderr")] {
type DiagnosticBufferInner = alloc::vec::Vec<u8>;
pub(crate) use std::io::Write as _ErrorWrite;
} else {
type DiagnosticBufferInner = String;
pub(crate) use core::fmt::Write as _ErrorWrite;
}
}
// Using this indirect export to avoid duplicating the expect(...) for all three cases above.
#[cfg_attr(
not(any(feature = "spv-in", feature = "glsl-in")),
expect(
unused_imports,
reason = "only need `ErrorWrite` with an appropriate front-end."
)
)]
pub(crate) use _ErrorWrite as ErrorWrite;
pub(crate) struct DiagnosticBuffer {
inner: DiagnosticBufferInner,
}
impl DiagnosticBuffer {
#[cfg_attr(
not(feature = "termcolor"),
expect(
clippy::missing_const_for_fn,
reason = "`NoColor::new` isn't `const`, but other `inner`s are."
)
)]
pub fn new() -> Self {
cfg_if::cfg_if! {
if #[cfg(feature = "termcolor")] {
let inner = codespan_reporting::term::termcolor::NoColor::new(alloc::vec::Vec::new());
} else if #[cfg(feature = "stderr")] {
let inner = alloc::vec::Vec::new();
} else {
let inner = String::new();
}
};
Self { inner }
}
pub fn inner_mut(&mut self) -> &mut DiagnosticBufferInner {
&mut self.inner
}
pub fn into_string(self) -> String {
let Self { inner } = self;
cfg_if::cfg_if! {
if #[cfg(feature = "termcolor")] {
String::from_utf8(inner.into_inner()).unwrap()
} else if #[cfg(feature = "stderr")] {
String::from_utf8(inner).unwrap()
} else {
inner
}
}
}
}
impl<E> Error for ShaderError<E>
where
ShaderError<E>: fmt::Display,
E: Error + 'static,
{
fn source(&self) -> Option<&(dyn Error + 'static)> {
Some(&self.inner)
}
}
pub(crate) fn replace_control_chars(s: &str) -> Cow<'_, str> {
const REPLACEMENT_CHAR: &str = "\u{FFFD}";
debug_assert_eq!(
REPLACEMENT_CHAR.chars().next().unwrap(),
char::REPLACEMENT_CHARACTER
);
let mut res = Cow::Borrowed(s);
let mut offset = 0;
while let Some(found_pos) = res[offset..].find(|c: char| c.is_control() && !c.is_whitespace()) {
offset += found_pos;
let found_len = res[offset..].chars().next().unwrap().len_utf8();
res.to_mut()
.replace_range(offset..offset + found_len, REPLACEMENT_CHAR);
offset += REPLACEMENT_CHAR.len();
}
res
}
#[test]
fn test_replace_control_chars() {
// The UTF-8 encoding of \u{0080} is multiple bytes.
let input = "Foo\u{0080}Bar\u{0001}Baz\n";
let expected = "Foo\u{FFFD}Bar\u{FFFD}Baz\n";
assert_eq!(replace_control_chars(input), expected);
}