gl/gl_filter: Add GLFilterImpl with configurable filter/filter_texture

GLFilter should override either .filter() or .filter_texture(), not
both. This allows an implementer to pick at runtime what function is
overridden.
This commit is contained in:
Marijn Suijten 2020-08-28 20:30:46 +02:00 committed by Sebastian Dröge
parent 2e85ebe789
commit 0d338a003e
2 changed files with 368 additions and 0 deletions

View file

@ -0,0 +1,364 @@
use ffi::{GstGLFilter, GstGLFilterClass, GstGLMemory};
use gst::ffi::GstBuffer;
use super::prelude::*;
use crate::prelude::*;
use glib::translate::*;
use gst::subclass::prelude::*;
use gst::{result_from_gboolean, Buffer, Caps, LoggableError, PadDirection, CAT_RUST};
use crate::GLBaseFilter;
use crate::GLFilter;
use crate::GLMemory;
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
pub enum GLFilterMode {
Buffer,
Texture,
}
pub trait GLFilterImpl: GLFilterImplExt + GLBaseFilterImpl {
const MODE: GLFilterMode;
fn set_caps(
&self,
filter: &Self::Type,
incaps: &Caps,
outcaps: &Caps,
) -> Result<(), LoggableError> {
GLFilterImplExt::parent_set_caps(self, filter, incaps, outcaps)
}
fn filter(
&self,
filter: &Self::Type,
input: &Buffer,
output: &Buffer,
) -> Result<(), LoggableError> {
self.parent_filter(filter, input, output)
}
fn filter_texture(
&self,
filter: &Self::Type,
input: &GLMemory,
output: &GLMemory,
) -> Result<(), LoggableError> {
self.parent_filter_texture(filter, input, output)
}
fn init_fbo(&self, filter: &Self::Type) -> Result<(), LoggableError> {
self.parent_init_fbo(filter)
}
fn transform_internal_caps(
&self,
filter: &Self::Type,
direction: PadDirection,
caps: &Caps,
filter_caps: Option<&Caps>,
) -> Option<Caps> {
self.parent_transform_internal_caps(filter, direction, caps, filter_caps)
}
}
pub trait GLFilterImplExt: ObjectSubclass {
fn parent_set_caps(
&self,
filter: &Self::Type,
incaps: &Caps,
outcaps: &Caps,
) -> Result<(), LoggableError>;
fn parent_filter(
&self,
filter: &Self::Type,
input: &Buffer,
output: &Buffer,
) -> Result<(), LoggableError>;
fn parent_filter_texture(
&self,
filter: &Self::Type,
input: &GLMemory,
output: &GLMemory,
) -> Result<(), LoggableError>;
fn parent_init_fbo(&self, filter: &Self::Type) -> Result<(), LoggableError>;
fn parent_transform_internal_caps(
&self,
filter: &Self::Type,
direction: PadDirection,
caps: &Caps,
filter_caps: Option<&Caps>,
) -> Option<Caps>;
}
impl<T: GLFilterImpl> GLFilterImplExt for T {
fn parent_set_caps(
&self,
filter: &Self::Type,
incaps: &Caps,
outcaps: &Caps,
) -> Result<(), LoggableError> {
unsafe {
let data = T::type_data();
let parent_class = data.as_ref().parent_class() as *mut GstGLFilterClass;
(*parent_class)
.set_caps
.map(|f| {
result_from_gboolean!(
f(
filter.unsafe_cast_ref::<GLFilter>().to_glib_none().0,
incaps.to_glib_none().0,
outcaps.to_glib_none().0,
),
CAT_RUST,
"Parent function `set_caps` failed"
)
})
.unwrap_or(Ok(()))
}
}
fn parent_filter(
&self,
filter: &Self::Type,
input: &Buffer,
output: &Buffer,
) -> Result<(), LoggableError> {
unsafe {
let data = T::type_data();
let parent_class = data.as_ref().parent_class() as *mut GstGLFilterClass;
(*parent_class)
.filter
.map(|f| {
result_from_gboolean!(
f(
filter.unsafe_cast_ref::<GLFilter>().to_glib_none().0,
input.to_glib_none().0,
output.to_glib_none().0,
),
CAT_RUST,
"Parent function `filter` failed"
)
})
.unwrap_or(Ok(()))
}
}
fn parent_filter_texture(
&self,
filter: &Self::Type,
input: &GLMemory,
output: &GLMemory,
) -> Result<(), LoggableError> {
unsafe {
let data = T::type_data();
let parent_class = data.as_ref().parent_class() as *mut GstGLFilterClass;
(*parent_class)
.filter_texture
.map(|f| {
result_from_gboolean!(
f(
filter.unsafe_cast_ref::<GLFilter>().to_glib_none().0,
input.to_glib_none().0,
output.to_glib_none().0,
),
CAT_RUST,
"Parent function `filter_texture` failed"
)
})
.unwrap_or(Ok(()))
}
}
fn parent_init_fbo(&self, filter: &Self::Type) -> Result<(), LoggableError> {
unsafe {
let data = T::type_data();
let parent_class = data.as_ref().parent_class() as *mut GstGLFilterClass;
(*parent_class)
.init_fbo
.map(|f| {
result_from_gboolean!(
f(filter.unsafe_cast_ref::<GLFilter>().to_glib_none().0),
CAT_RUST,
"Parent function `init_fbo` failed"
)
})
.unwrap_or(Ok(()))
}
}
fn parent_transform_internal_caps(
&self,
filter: &Self::Type,
direction: PadDirection,
caps: &Caps,
filter_caps: Option<&Caps>,
) -> Option<Caps> {
unsafe {
let data = T::type_data();
let parent_class = data.as_ref().parent_class() as *mut GstGLFilterClass;
let f = (*parent_class)
.transform_internal_caps
.expect("Missing parent function `transform_internal_caps`");
from_glib_full(f(
filter.unsafe_cast_ref::<GLFilter>().to_glib_none().0,
direction.to_glib(),
caps.to_glib_none().0,
filter_caps.to_glib_none().0,
))
}
}
}
unsafe impl<T: GLFilterImpl> IsSubclassable<T> for GLFilter {
fn class_init(klass: &mut glib::Class<Self>) {
<GLBaseFilter as IsSubclassable<T>>::class_init(klass);
let klass = klass.as_mut();
klass.set_caps = Some(set_caps::<T>);
klass.init_fbo = Some(init_fbo::<T>);
klass.transform_internal_caps = Some(transform_internal_caps::<T>);
match <T as GLFilterImpl>::MODE {
GLFilterMode::Buffer => {
klass.filter = Some(filter::<T>);
klass.filter_texture = None;
}
GLFilterMode::Texture => {
klass.filter = None;
klass.filter_texture = Some(filter_texture::<T>);
}
}
}
fn instance_init(instance: &mut glib::subclass::InitializingObject<T>) {
<GLBaseFilter as IsSubclassable<T>>::instance_init(instance)
}
}
unsafe extern "C" fn filter<T: GLFilterImpl>(
ptr: *mut GstGLFilter,
input: *mut GstBuffer,
output: *mut GstBuffer,
) -> glib::ffi::gboolean {
let instance = &*(ptr as *mut T::Instance);
let imp = instance.impl_();
let wrap: Borrowed<GLFilter> = from_glib_borrow(ptr);
gst::panic_to_error!(&wrap, &imp.panicked(), false, {
match imp.filter(
wrap.unsafe_cast_ref(),
&from_glib_borrow(input),
&from_glib_borrow(output),
) {
Ok(()) => true,
Err(err) => {
err.log_with_object(&*wrap);
false
}
}
})
.to_glib()
}
unsafe extern "C" fn filter_texture<T: GLFilterImpl>(
ptr: *mut GstGLFilter,
input: *mut GstGLMemory,
output: *mut GstGLMemory,
) -> glib::ffi::gboolean {
let instance = &*(ptr as *mut T::Instance);
let imp = instance.impl_();
let wrap: Borrowed<GLFilter> = from_glib_borrow(ptr);
gst::panic_to_error!(&wrap, &imp.panicked(), false, {
match imp.filter_texture(
wrap.unsafe_cast_ref(),
&from_glib_borrow(input),
&from_glib_borrow(output),
) {
Ok(()) => true,
Err(err) => {
err.log_with_object(&*wrap);
false
}
}
})
.to_glib()
}
unsafe extern "C" fn init_fbo<T: GLFilterImpl>(ptr: *mut GstGLFilter) -> glib::ffi::gboolean {
let instance = &*(ptr as *mut T::Instance);
let imp = instance.impl_();
let wrap: Borrowed<GLFilter> = from_glib_borrow(ptr);
gst::panic_to_error!(&wrap, &imp.panicked(), false, {
match imp.init_fbo(wrap.unsafe_cast_ref()) {
Ok(()) => true,
Err(err) => {
err.log_with_object(&*wrap);
false
}
}
})
.to_glib()
}
unsafe extern "C" fn set_caps<T: GLFilterImpl>(
ptr: *mut GstGLFilter,
incaps: *mut gst::ffi::GstCaps,
outcaps: *mut gst::ffi::GstCaps,
) -> glib::ffi::gboolean {
let instance = &*(ptr as *mut T::Instance);
let imp = instance.impl_();
let wrap: Borrowed<GLFilter> = from_glib_borrow(ptr);
gst::panic_to_error!(&wrap, &imp.panicked(), false, {
match GLFilterImpl::set_caps(
imp,
wrap.unsafe_cast_ref(),
&from_glib_borrow(incaps),
&from_glib_borrow(outcaps),
) {
Ok(()) => true,
Err(err) => {
err.log_with_object(&*wrap);
false
}
}
})
.to_glib()
}
unsafe extern "C" fn transform_internal_caps<T: GLFilterImpl>(
ptr: *mut GstGLFilter,
direction: gst::ffi::GstPadDirection,
caps: *mut gst::ffi::GstCaps,
filter_caps: *mut gst::ffi::GstCaps,
) -> *mut gst::ffi::GstCaps {
let instance = &*(ptr as *mut T::Instance);
let imp = instance.impl_();
let wrap: Borrowed<GLFilter> = from_glib_borrow(ptr);
gst::panic_to_error!(&wrap, &imp.panicked(), None, {
let filter_caps: Borrowed<Option<Caps>> = from_glib_borrow(filter_caps);
imp.transform_internal_caps(
wrap.unsafe_cast_ref(),
from_glib(direction),
&from_glib_borrow(caps),
filter_caps.as_ref().as_ref(),
)
})
.map(|caps| caps.into_ptr())
.unwrap_or(std::ptr::null_mut())
}

View file

@ -1,5 +1,9 @@
mod gl_base_filter;
mod gl_filter;
pub use self::gl_filter::GLFilterMode;
pub mod prelude {
pub use super::gl_base_filter::{GLBaseFilterImpl, GLBaseFilterImplExt};
pub use super::gl_filter::{GLFilterImpl, GLFilterImplExt};
}