gstreamer_audio/subclass/
audio_aggregator.rs1use std::ptr;
4
5use glib::translate::*;
6use gst_base::{prelude::*, subclass::prelude::*};
7
8use crate::{AudioAggregator, AudioAggregatorPad, ffi};
9
10pub trait AudioAggregatorImpl: AggregatorImpl + ObjectSubclass<Type: IsA<AudioAggregator>> {
11 fn create_output_buffer(&self, num_frames: u32) -> Option<gst::Buffer> {
13 self.parent_create_output_buffer(num_frames)
14 }
15
16 #[allow(clippy::too_many_arguments)]
21 fn aggregate_one_buffer(
22 &self,
23 pad: &AudioAggregatorPad,
24 inbuf: &gst::BufferRef,
25 in_offset: u32,
26 outbuf: &mut gst::BufferRef,
27 out_offset: u32,
28 num_frames: u32,
29 ) -> bool {
30 self.parent_aggregate_one_buffer(pad, inbuf, in_offset, outbuf, out_offset, num_frames)
31 }
32}
33
34pub trait AudioAggregatorImplExt: AudioAggregatorImpl {
35 fn parent_create_output_buffer(&self, num_frames: u32) -> Option<gst::Buffer> {
36 unsafe {
37 let data = Self::type_data();
38 let parent_class = data.as_ref().parent_class() as *mut ffi::GstAudioAggregatorClass;
39 let f = (*parent_class)
40 .create_output_buffer
41 .expect("Missing parent function `create_output_buffer`");
42
43 from_glib_full(f(
44 self.obj()
45 .unsafe_cast_ref::<AudioAggregator>()
46 .to_glib_none()
47 .0,
48 num_frames,
49 ))
50 }
51 }
52
53 fn parent_aggregate_one_buffer(
54 &self,
55 pad: &AudioAggregatorPad,
56 inbuf: &gst::BufferRef,
57 in_offset: u32,
58 outbuf: &mut gst::BufferRef,
59 out_offset: u32,
60 num_frames: u32,
61 ) -> bool {
62 unsafe {
63 let data = Self::type_data();
64 let parent_class = data.as_ref().parent_class() as *mut ffi::GstAudioAggregatorClass;
65 let f = (*parent_class)
66 .aggregate_one_buffer
67 .expect("Missing parent function `aggregate_one_buffer`");
68
69 from_glib(f(
70 self.obj()
71 .unsafe_cast_ref::<AudioAggregator>()
72 .to_glib_none()
73 .0,
74 pad.to_glib_none().0,
75 inbuf.as_mut_ptr(),
76 in_offset,
77 outbuf.as_mut_ptr(),
78 out_offset,
79 num_frames,
80 ))
81 }
82 }
83}
84
85impl<T: AudioAggregatorImpl> AudioAggregatorImplExt for T {}
86
87unsafe impl<T: AudioAggregatorImpl> IsSubclassable<T> for AudioAggregator {
88 fn class_init(klass: &mut glib::Class<Self>) {
89 Self::parent_class_init::<T>(klass);
90
91 let klass = klass.as_mut();
92 klass.create_output_buffer = Some(audio_aggregator_create_output_buffer::<T>);
93 klass.aggregate_one_buffer = Some(audio_aggregator_aggregate_one_buffer::<T>);
94 }
95}
96
97unsafe extern "C" fn audio_aggregator_create_output_buffer<T: AudioAggregatorImpl>(
98 ptr: *mut ffi::GstAudioAggregator,
99 num_frames: u32,
100) -> *mut gst::ffi::GstBuffer {
101 unsafe {
102 let instance = &*(ptr as *mut T::Instance);
103 let imp = instance.imp();
104
105 gst::panic_to_error!(imp, None, { imp.create_output_buffer(num_frames) })
106 .map(|buffer| buffer.into_glib_ptr())
107 .unwrap_or(ptr::null_mut())
108 }
109}
110
111unsafe extern "C" fn audio_aggregator_aggregate_one_buffer<T: AudioAggregatorImpl>(
112 ptr: *mut ffi::GstAudioAggregator,
113 pad: *mut ffi::GstAudioAggregatorPad,
114 inbuf: *mut gst::ffi::GstBuffer,
115 in_offset: u32,
116 outbuf: *mut gst::ffi::GstBuffer,
117 out_offset: u32,
118 num_frames: u32,
119) -> glib::ffi::gboolean {
120 unsafe {
121 let instance = &*(ptr as *mut T::Instance);
122 let imp = instance.imp();
123
124 gst::panic_to_error!(imp, true, {
125 imp.aggregate_one_buffer(
126 &from_glib_borrow(pad),
127 gst::BufferRef::from_ptr(inbuf),
128 in_offset,
129 gst::BufferRef::from_mut_ptr(outbuf),
130 out_offset,
131 num_frames,
132 )
133 })
134 .into_glib()
135 }
136}