244 lines
9.1 KiB
Rust
244 lines
9.1 KiB
Rust
//! librashader error C API. (`libra_error_*`).
|
|
use std::any::Any;
|
|
use std::ffi::{c_char, CString};
|
|
use std::mem::MaybeUninit;
|
|
use std::ptr::NonNull;
|
|
use thiserror::Error;
|
|
|
|
/// The error type for librashader C API.
|
|
#[non_exhaustive]
|
|
#[derive(Error, Debug)]
|
|
pub enum LibrashaderError {
|
|
#[error("There was an unknown error.")]
|
|
UnknownError(Box<dyn Any + Send + 'static>),
|
|
#[error("The parameter was null or invalid.")]
|
|
InvalidParameter(&'static str),
|
|
#[error("The provided string was not valid UTF8.")]
|
|
InvalidString(#[from] std::str::Utf8Error),
|
|
#[error("There was an error parsing the preset.")]
|
|
PresetError(#[from] librashader::presets::ParsePresetError),
|
|
#[error("There was an error preprocessing the shader source.")]
|
|
PreprocessError(#[from] librashader::preprocess::PreprocessError),
|
|
#[error("There was an error compiling the shader source.")]
|
|
ShaderCompileError(#[from] librashader::reflect::ShaderCompileError),
|
|
#[error("There was an error reflecting the shader source.")]
|
|
ShaderReflectError(#[from] librashader::reflect::ShaderReflectError),
|
|
#[error("The provided parameter name was invalid.")]
|
|
UnknownShaderParameter(*const c_char),
|
|
#[cfg(feature = "runtime-opengl")]
|
|
#[doc(cfg(feature = "runtime-opengl"))]
|
|
#[error("There was an error in the OpenGL filter chain.")]
|
|
OpenGlFilterError(#[from] librashader::runtime::gl::error::FilterChainError),
|
|
#[cfg(all(target_os = "windows", feature = "runtime-d3d11"))]
|
|
#[doc(cfg(all(target_os = "windows", feature = "runtime-d3d11")))]
|
|
#[error("There was an error in the D3D11 filter chain.")]
|
|
D3D11FilterError(#[from] librashader::runtime::d3d11::error::FilterChainError),
|
|
#[cfg(all(target_os = "windows", feature = "runtime-d3d12"))]
|
|
#[doc(cfg(all(target_os = "windows", feature = "runtime-d3d12")))]
|
|
#[error("There was an error in the D3D12 filter chain.")]
|
|
D3D12FilterError(#[from] librashader::runtime::d3d12::error::FilterChainError),
|
|
#[cfg(feature = "runtime-vulkan")]
|
|
#[doc(cfg(feature = "runtime-vulkan"))]
|
|
#[error("There was an error in the Vulkan filter chain.")]
|
|
VulkanFilterError(#[from] librashader::runtime::vk::error::FilterChainError),
|
|
#[doc(cfg(all(target_vendor = "apple", feature = "runtime-metal")))]
|
|
#[cfg(all(target_vendor = "apple", feature = "runtime-metal"))]
|
|
#[error("There was an error in the D3D12 filter chain.")]
|
|
MetalFilterError(#[from] librashader::runtime::mtl::error::FilterChainError),
|
|
}
|
|
|
|
/// Error codes for librashader error types.
|
|
#[repr(i32)]
|
|
pub enum LIBRA_ERRNO {
|
|
UNKNOWN_ERROR = 0,
|
|
INVALID_PARAMETER = 1,
|
|
INVALID_STRING = 2,
|
|
PRESET_ERROR = 3,
|
|
PREPROCESS_ERROR = 4,
|
|
SHADER_PARAMETER_ERROR = 5,
|
|
REFLECT_ERROR = 6,
|
|
RUNTIME_ERROR = 7,
|
|
}
|
|
|
|
// Nothing here can use extern_fn because they are lower level than libra_error_t.
|
|
|
|
/// Function pointer definition for libra_error_errno
|
|
pub type PFN_libra_error_errno = extern "C" fn(error: libra_error_t) -> LIBRA_ERRNO;
|
|
#[no_mangle]
|
|
/// Get the error code corresponding to this error object.
|
|
///
|
|
/// ## Safety
|
|
/// - `error` must be valid and initialized.
|
|
pub unsafe extern "C" fn libra_error_errno(error: libra_error_t) -> LIBRA_ERRNO {
|
|
let Some(error) = error else {
|
|
return LIBRA_ERRNO::UNKNOWN_ERROR;
|
|
};
|
|
|
|
unsafe { error.as_ref().get_code() }
|
|
}
|
|
|
|
/// Function pointer definition for libra_error_print
|
|
pub type PFN_libra_error_print = extern "C" fn(error: libra_error_t) -> i32;
|
|
#[no_mangle]
|
|
/// Print the error message.
|
|
///
|
|
/// If `error` is null, this function does nothing and returns 1. Otherwise, this function returns 0.
|
|
/// ## Safety
|
|
/// - `error` must be a valid and initialized instance of `libra_error_t`.
|
|
pub unsafe extern "C" fn libra_error_print(error: libra_error_t) -> i32 {
|
|
let Some(error) = error else { return 1 };
|
|
unsafe {
|
|
let error = error.as_ref();
|
|
println!("{error:?}: {error}");
|
|
}
|
|
0
|
|
}
|
|
|
|
/// Function pointer definition for libra_error_free
|
|
pub type PFN_libra_error_free = extern "C" fn(error: *mut libra_error_t) -> i32;
|
|
#[no_mangle]
|
|
/// Frees any internal state kept by the error.
|
|
///
|
|
/// If `error` is null, this function does nothing and returns 1. Otherwise, this function returns 0.
|
|
/// The resulting error object becomes null.
|
|
/// ## Safety
|
|
/// - `error` must be null or a pointer to a valid and initialized instance of `libra_error_t`.
|
|
pub unsafe extern "C" fn libra_error_free(error: *mut libra_error_t) -> i32 {
|
|
if error.is_null() {
|
|
return 1;
|
|
}
|
|
|
|
let error = unsafe { &mut *error };
|
|
let error = error.take();
|
|
let Some(error) = error else {
|
|
return 1;
|
|
};
|
|
|
|
unsafe { drop(Box::from_raw(error.as_ptr())) }
|
|
0
|
|
}
|
|
|
|
/// Function pointer definition for libra_error_write
|
|
pub type PFN_libra_error_write =
|
|
extern "C" fn(error: libra_error_t, out: *mut MaybeUninit<*mut c_char>) -> i32;
|
|
#[no_mangle]
|
|
/// Writes the error message into `out`
|
|
///
|
|
/// If `error` is null, this function does nothing and returns 1. Otherwise, this function returns 0.
|
|
/// ## Safety
|
|
/// - `error` must be a valid and initialized instance of `libra_error_t`.
|
|
/// - `out` must be a non-null pointer. The resulting string must not be modified.
|
|
pub unsafe extern "C" fn libra_error_write(
|
|
error: libra_error_t,
|
|
out: *mut MaybeUninit<*mut c_char>,
|
|
) -> i32 {
|
|
let Some(error) = error else { return 1 };
|
|
if out.is_null() {
|
|
return 1;
|
|
}
|
|
|
|
unsafe {
|
|
let error = error.as_ref();
|
|
let Ok(cstring) = CString::new(format!("{error:?}: {error}")) else {
|
|
return 1;
|
|
};
|
|
|
|
out.write(MaybeUninit::new(cstring.into_raw()))
|
|
}
|
|
0
|
|
}
|
|
|
|
/// Function pointer definition for libra_error_free_string
|
|
pub type PFN_libra_error_free_string = extern "C" fn(out: *mut *mut c_char) -> i32;
|
|
#[no_mangle]
|
|
/// Frees an error string previously allocated by `libra_error_write`.
|
|
///
|
|
/// After freeing, the pointer will be set to null.
|
|
/// ## Safety
|
|
/// - If `libra_error_write` is not null, it must point to a string previously returned by `libra_error_write`.
|
|
/// Attempting to free anything else, including strings or objects from other librashader functions, is immediate
|
|
/// Undefined Behaviour.
|
|
pub unsafe extern "C" fn libra_error_free_string(out: *mut *mut c_char) -> i32 {
|
|
if out.is_null() {
|
|
return 1;
|
|
}
|
|
|
|
unsafe {
|
|
let ptr = out.read();
|
|
*out = std::ptr::null_mut();
|
|
drop(CString::from_raw(ptr))
|
|
}
|
|
0
|
|
}
|
|
|
|
impl LibrashaderError {
|
|
pub(crate) const fn get_code(&self) -> LIBRA_ERRNO {
|
|
match self {
|
|
LibrashaderError::UnknownError(_) => LIBRA_ERRNO::UNKNOWN_ERROR,
|
|
LibrashaderError::InvalidParameter(_) => LIBRA_ERRNO::INVALID_PARAMETER,
|
|
LibrashaderError::InvalidString(_) => LIBRA_ERRNO::INVALID_STRING,
|
|
LibrashaderError::PresetError(_) => LIBRA_ERRNO::PRESET_ERROR,
|
|
LibrashaderError::PreprocessError(_) => LIBRA_ERRNO::PREPROCESS_ERROR,
|
|
LibrashaderError::ShaderCompileError(_) | LibrashaderError::ShaderReflectError(_) => {
|
|
LIBRA_ERRNO::REFLECT_ERROR
|
|
}
|
|
LibrashaderError::UnknownShaderParameter(_) => LIBRA_ERRNO::SHADER_PARAMETER_ERROR,
|
|
#[cfg(feature = "runtime-opengl")]
|
|
LibrashaderError::OpenGlFilterError(_) => LIBRA_ERRNO::RUNTIME_ERROR,
|
|
#[cfg(all(target_os = "windows", feature = "runtime-d3d11"))]
|
|
LibrashaderError::D3D11FilterError(_) => LIBRA_ERRNO::RUNTIME_ERROR,
|
|
#[cfg(all(target_os = "windows", feature = "runtime-d3d12"))]
|
|
LibrashaderError::D3D12FilterError(_) => LIBRA_ERRNO::RUNTIME_ERROR,
|
|
#[cfg(feature = "runtime-vulkan")]
|
|
LibrashaderError::VulkanFilterError(_) => LIBRA_ERRNO::RUNTIME_ERROR,
|
|
#[cfg(all(target_vendor = "apple", feature = "runtime-metal"))]
|
|
LibrashaderError::MetalFilterError(_) => LIBRA_ERRNO::RUNTIME_ERROR,
|
|
}
|
|
}
|
|
pub(crate) const fn ok() -> libra_error_t {
|
|
None
|
|
}
|
|
|
|
pub(crate) fn export(self) -> libra_error_t {
|
|
NonNull::new(Box::into_raw(Box::new(self)))
|
|
}
|
|
}
|
|
|
|
macro_rules! assert_non_null {
|
|
($value:ident) => {
|
|
if $value.is_null() || !$value.is_aligned() {
|
|
return $crate::error::LibrashaderError::InvalidParameter(stringify!($value)).export();
|
|
}
|
|
};
|
|
(noexport $value:ident) => {
|
|
if $value.is_null() || !$value.is_aligned() {
|
|
return Err($crate::error::LibrashaderError::InvalidParameter(
|
|
stringify!($value),
|
|
));
|
|
}
|
|
};
|
|
}
|
|
|
|
macro_rules! assert_some_ptr {
|
|
($value:ident) => {
|
|
if $value.is_none() {
|
|
return $crate::error::LibrashaderError::InvalidParameter(stringify!($value)).export();
|
|
}
|
|
|
|
let $value = unsafe { $value.as_ref().unwrap_unchecked().as_ref() };
|
|
};
|
|
(mut $value:ident) => {
|
|
if $value.is_none() {
|
|
return $crate::error::LibrashaderError::InvalidParameter(stringify!($value)).export();
|
|
}
|
|
|
|
let $value = unsafe { $value.as_mut().unwrap_unchecked().as_mut() };
|
|
};
|
|
}
|
|
|
|
use crate::ctypes::libra_error_t;
|
|
pub(crate) use assert_non_null;
|
|
|
|
// pub(crate) use assert_some;
|
|
pub(crate) use assert_some_ptr;
|