You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
160 lines
4.5 KiB
160 lines
4.5 KiB
//! Functionality for parsing and using PICA200 shaders on the 3DS. This module |
|
//! does not compile shaders, but enables using pre-compiled shaders at runtime. |
|
//! |
|
//! For more details about the PICA200 compiler / shader language, see |
|
//! documentation for <https://github.com/devkitPro/picasso>. |
|
|
|
use std::error::Error; |
|
use std::mem::MaybeUninit; |
|
|
|
pub mod macros; |
|
|
|
/// A PICA200 shader program. It may have one or both of: |
|
/// |
|
/// * A vertex [shader instance](Instance) |
|
/// * A geometry [shader instance](Instance) |
|
/// |
|
/// The PICA200 does not support user-programmable fragment shaders. |
|
pub struct Program { |
|
program: citro3d_sys::shaderProgram_s, |
|
} |
|
|
|
impl Program { |
|
/// Create a new shader program from a vertex shader. |
|
/// |
|
/// # Errors |
|
/// |
|
/// Returns an error if: |
|
/// * the shader program cannot be initialized |
|
/// * the input shader is not a vertex shader or is otherwise invalid |
|
pub fn new(vertex_shader: Entrypoint) -> Result<Self, ctru::Error> { |
|
let mut program = unsafe { |
|
let mut program = MaybeUninit::uninit(); |
|
let result = citro3d_sys::shaderProgramInit(program.as_mut_ptr()); |
|
if result != 0 { |
|
return Err(ctru::Error::from(result)); |
|
} |
|
program.assume_init() |
|
}; |
|
|
|
let ret = unsafe { citro3d_sys::shaderProgramSetVsh(&mut program, vertex_shader.as_raw()) }; |
|
|
|
if ret == 0 { |
|
Ok(Self { program }) |
|
} else { |
|
Err(ctru::Error::from(ret)) |
|
} |
|
} |
|
|
|
/// Set the geometry shader for a given program. |
|
/// |
|
/// # Errors |
|
/// |
|
/// Returns an error if the input shader is not a geometry shader or is |
|
/// otherwise invalid. |
|
pub fn set_geometry_shader( |
|
&mut self, |
|
geometry_shader: Entrypoint, |
|
stride: u8, |
|
) -> Result<(), ctru::Error> { |
|
let ret = unsafe { |
|
citro3d_sys::shaderProgramSetGsh(&mut self.program, geometry_shader.as_raw(), stride) |
|
}; |
|
|
|
if ret == 0 { |
|
Ok(()) |
|
} else { |
|
Err(ctru::Error::from(ret)) |
|
} |
|
} |
|
|
|
// TODO: pub(crate) |
|
pub fn as_raw(&mut self) -> *mut citro3d_sys::shaderProgram_s { |
|
&mut self.program |
|
} |
|
} |
|
|
|
impl Drop for Program { |
|
fn drop(&mut self) { |
|
unsafe { |
|
let _ = citro3d_sys::shaderProgramFree(self.as_raw()); |
|
} |
|
} |
|
} |
|
|
|
/// A PICA200 Shader Library (commonly called DVLB). This can be comprised of |
|
/// one or more [`Entrypoint`]s, but most commonly has one vertex shader and an |
|
/// optional geometry shader. |
|
/// |
|
/// This is the result of parsing a shader binary (shbin), and the resulting |
|
/// [`Entrypoint`]s can be used as part of a [`Program`]. |
|
pub struct Library(*mut citro3d_sys::DVLB_s); |
|
|
|
impl Library { |
|
/// Parse a new shader library from input bytes. |
|
/// |
|
/// # Errors |
|
/// |
|
/// An error is returned if the input data does not have an alignment of 4 |
|
/// (cannot be safely converted to `&[u32]`). |
|
pub fn from_bytes(bytes: &[u8]) -> Result<Self, Box<dyn Error>> { |
|
let aligned: &[u32] = bytemuck::try_cast_slice(bytes)?; |
|
Ok(Self(unsafe { |
|
citro3d_sys::DVLB_ParseFile( |
|
// SAFETY: we're trusting the parse implementation doesn't mutate |
|
// the contents of the data. From a quick read it looks like that's |
|
// correct and it should just take a const arg in the API. |
|
aligned.as_ptr() as *mut _, |
|
aligned.len().try_into()?, |
|
) |
|
})) |
|
} |
|
|
|
#[must_use] |
|
pub fn len(&self) -> usize { |
|
unsafe { (*self.0).numDVLE as usize } |
|
} |
|
|
|
#[must_use] |
|
pub fn get(&self, index: usize) -> Option<Entrypoint> { |
|
if index < self.len() { |
|
Some(Entrypoint { |
|
ptr: unsafe { (*self.0).DVLE.add(index) }, |
|
_library: self, |
|
}) |
|
} else { |
|
None |
|
} |
|
} |
|
|
|
#[must_use] |
|
pub fn is_empty(&self) -> bool { |
|
self.len() == 0 |
|
} |
|
|
|
fn as_raw(&mut self) -> *mut citro3d_sys::DVLB_s { |
|
self.0 |
|
} |
|
} |
|
|
|
impl Drop for Library { |
|
fn drop(&mut self) { |
|
unsafe { |
|
citro3d_sys::DVLB_Free(self.as_raw()); |
|
} |
|
} |
|
} |
|
|
|
/// A shader library entrypoint (also called DVLE). This represents either a |
|
/// vertex or a geometry shader. |
|
#[derive(Clone, Copy)] |
|
pub struct Entrypoint<'lib> { |
|
ptr: *mut citro3d_sys::DVLE_s, |
|
_library: &'lib Library, |
|
} |
|
|
|
impl<'lib> Entrypoint<'lib> { |
|
fn as_raw(self) -> *mut citro3d_sys::DVLE_s { |
|
self.ptr |
|
} |
|
}
|
|
|