pub use {format::*, mapping::*, vao::*};
pub type AttrArr<D> = ArrObject<Attribute, D>;
pub type IdxArr<D> = ArrObject<Index, D>;
pub type UniformArr<D> = ShdArrObj<Uniform, D>;
pub type ShdStorageArr<D> = ShdArrObj<ShdStorage, D>;
pub struct ShdArrObj<T: ShdBuffType, D> {
pub array: ArrObject<T, D>,
loc: Cell<u32>,
}
impl<T: ShdBuffType, D> ShdArrObj<T, D> {
pub fn new(args: impl AllocArgs<D>) -> Self {
ArrObject::new(args).into()
}
}
impl<T: ShdBuffType, D> Drop for ShdArrObj<T, D> {
fn drop(&mut self) {
UniformState::<T>::drop(self.array.obj);
}
}
impl<T: ShdBuffType, D> From<ArrObject<T, D>> for ShdArrObj<T, D> {
fn from(array: ArrObject<T, D>) -> Self {
let (size, max) = (array.size(), T::max_size());
if size > max {
FAIL!("GL {} buffer({}|{size}) exceeds maximum size {max}", type_name::<T>(), array.obj);
}
Self { array, loc: Def() }
}
}
impl<D> UniformArr<D> {
pub fn Bind(&self) -> ShdArrBinding<Uniform> {
let loc = self.loc.take();
let (b, l) = ShdArrBinding::<Uniform>::new(self, loc);
self.loc.set(l);
b
}
}
impl<D> ShdStorageArr<D> {
pub fn Bind(&self, loc: u32) -> Option<ShdArrBinding<ShdStorage>> {
ShdArrBinding::<ShdStorage>::new(self, loc)
}
}
pub struct ShdArrBinding<'l, T: ShdBuffType> {
t: Dummy<&'l T>,
pub l: u32,
}
impl<'l> ShdArrBinding<'l, Uniform> {
pub fn new<D>(o: &'l UniformArr<D>, hint: u32) -> (Self, u32) {
let l = UniformState::<Uniform>::Bind(o.array.obj, hint);
(Self { t: Dummy, l }, l)
}
}
impl<'l> ShdArrBinding<'l, ShdStorage> {
pub fn new<D>(o: &'l ShdStorageArr<D>, loc: u32) -> Option<Self> {
if UniformState::<ShdStorage>::BindLocation(o.array.obj, loc) {
Some(Self { t: Dummy, l: loc })
} else {
None
}
}
}
impl<T: ShdBuffType> Drop for ShdArrBinding<'_, T> {
fn drop(&mut self) {
UniformState::<T>::Unbind(self.l);
}
}
mod args;
mod format;
mod mapping;
mod vao;
mod vao_args;
use {super::internal::*, crate::lib::*, args::*};