use core::marker::PhantomData;
pub trait HasStage {
type AStage;
fn new_shader(shader_type: Type, src: &str) -> Result<Self::AStage, StageError>;
fn free_shader(shader: &mut Self::AStage);
}
pub trait ShaderTypeable {
fn shader_type() -> Type;
}
#[derive(Clone, Copy, Debug)]
pub enum Type {
TessellationControlShader,
TessellationEvaluationShader,
VertexShader,
GeometryShader,
FragmentShader
}
#[derive(Debug)]
pub struct TessellationControlShader;
impl ShaderTypeable for TessellationControlShader {
fn shader_type() -> Type { Type::TessellationControlShader }
}
#[derive(Debug)]
pub struct TessellationEvaluationShader;
impl ShaderTypeable for TessellationEvaluationShader {
fn shader_type() -> Type { Type::TessellationEvaluationShader }
}
#[derive(Debug)]
pub struct VertexShader;
impl ShaderTypeable for VertexShader {
fn shader_type() -> Type { Type::VertexShader }
}
#[derive(Debug)]
pub struct GeometryShader;
impl ShaderTypeable for GeometryShader {
fn shader_type() -> Type { Type::GeometryShader }
}
#[derive(Debug)]
pub struct FragmentShader;
impl ShaderTypeable for FragmentShader {
fn shader_type() -> Type { Type::FragmentShader }
}
#[derive(Debug)]
pub struct Stage<C, T> where C: HasStage {
pub repr: C::AStage,
_t: PhantomData<T>
}
impl<C, T> Drop for Stage<C, T> where C: HasStage {
fn drop(&mut self) {
C::free_shader(&mut self.repr)
}
}
impl<C, T> Stage<C, T> where C: HasStage, T: ShaderTypeable {
pub fn new(src: &str) -> Result<Self, StageError> {
let shader = C::new_shader(T::shader_type(), src);
shader.map(|shader| Stage {
repr: shader,
_t: PhantomData
})
}
}
#[derive(Clone, Debug)]
pub enum StageError {
CompilationFailed(Type, String),
UnsupportedType(Type)
}