forked from mirrors/gstreamer-rs
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:
parent
2e85ebe789
commit
0d338a003e
2 changed files with 368 additions and 0 deletions
364
gstreamer-gl/src/subclass/gl_filter.rs
Normal file
364
gstreamer-gl/src/subclass/gl_filter.rs
Normal 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())
|
||||||
|
}
|
|
@ -1,5 +1,9 @@
|
||||||
mod gl_base_filter;
|
mod gl_base_filter;
|
||||||
|
mod gl_filter;
|
||||||
|
|
||||||
|
pub use self::gl_filter::GLFilterMode;
|
||||||
|
|
||||||
pub mod prelude {
|
pub mod prelude {
|
||||||
pub use super::gl_base_filter::{GLBaseFilterImpl, GLBaseFilterImplExt};
|
pub use super::gl_base_filter::{GLBaseFilterImpl, GLBaseFilterImplExt};
|
||||||
|
pub use super::gl_filter::{GLFilterImpl, GLFilterImplExt};
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue