Allow changing bus sync handler and appsink/src callbacks when running with 1.16.3 or newer

Previously it was not thread-safe to change them and could lead to
crashes but with 1.16.3 it is now.

Unsetting the bus sync handler before 1.16.3 will have no effect at all,
setting a new bus sync handler or appsink/src callbacks will panic.

This partially reverts 2f88dc6576
This commit is contained in:
Sebastian Dröge 2020-02-15 10:05:25 +01:00
parent a2a1a87c46
commit d58cf01e3a
3 changed files with 85 additions and 13 deletions

View file

@ -190,12 +190,22 @@ impl AppSink {
unsafe { unsafe {
let sink = self.to_glib_none().0; let sink = self.to_glib_none().0;
if !gobject_sys::g_object_get_qdata(sink as *mut _, SET_ONCE_QUARK.to_glib()).is_null()
// This is not thread-safe before 1.16.3, see
// https://gitlab.freedesktop.org/gstreamer/gst-plugins-base/merge_requests/570
if gst::version() < (1, 16, 3, 0) {
if !gobject_sys::g_object_get_qdata(sink as *mut _, SET_ONCE_QUARK.to_glib())
.is_null()
{ {
panic!("AppSink callbacks can only be set once"); panic!("AppSink callbacks can only be set once");
} }
gobject_sys::g_object_set_qdata(sink as *mut _, SET_ONCE_QUARK.to_glib(), 1 as *mut _); gobject_sys::g_object_set_qdata(
sink as *mut _,
SET_ONCE_QUARK.to_glib(),
1 as *mut _,
);
}
gst_app_sys::gst_app_sink_set_callbacks( gst_app_sys::gst_app_sink_set_callbacks(
sink, sink,
@ -315,6 +325,17 @@ impl AppSinkStream {
} }
} }
#[cfg(any(feature = "v1_10"))]
impl Drop for AppSinkStream {
fn drop(&mut self) {
// This is not thread-safe before 1.16.3, see
// https://gitlab.freedesktop.org/gstreamer/gst-plugins-base/merge_requests/570
if gst::version() >= (1, 16, 3, 0) {
self.app_sink.set_callbacks(AppSinkCallbacks::new().build());
}
}
}
#[cfg(any(feature = "v1_10"))] #[cfg(any(feature = "v1_10"))]
impl Stream for AppSinkStream { impl Stream for AppSinkStream {
type Item = gst::Sample; type Item = gst::Sample;

View file

@ -205,11 +205,21 @@ impl AppSrc {
unsafe { unsafe {
let src = self.to_glib_none().0; let src = self.to_glib_none().0;
if !gobject_sys::g_object_get_qdata(src as *mut _, SET_ONCE_QUARK.to_glib()).is_null() { // This is not thread-safe before 1.16.3, see
// https://gitlab.freedesktop.org/gstreamer/gst-plugins-base/merge_requests/570
if gst::version() < (1, 16, 3, 0) {
if !gobject_sys::g_object_get_qdata(src as *mut _, SET_ONCE_QUARK.to_glib())
.is_null()
{
panic!("AppSrc callbacks can only be set once"); panic!("AppSrc callbacks can only be set once");
} }
gobject_sys::g_object_set_qdata(src as *mut _, SET_ONCE_QUARK.to_glib(), 1 as *mut _); gobject_sys::g_object_set_qdata(
src as *mut _,
SET_ONCE_QUARK.to_glib(),
1 as *mut _,
);
}
gst_app_sys::gst_app_src_set_callbacks( gst_app_sys::gst_app_src_set_callbacks(
src, src,
@ -282,6 +292,16 @@ impl AppSrcSink {
} }
} }
impl Drop for AppSrcSink {
fn drop(&mut self) {
// This is not thread-safe before 1.16.3, see
// https://gitlab.freedesktop.org/gstreamer/gst-plugins-base/merge_requests/570
if gst::version() >= (1, 16, 3, 0) {
self.app_src.set_callbacks(AppSrcCallbacks::new().build());
}
}
}
impl Sink<gst::Sample> for AppSrcSink { impl Sink<gst::Sample> for AppSrcSink {
type Error = gst::FlowError; type Error = gst::FlowError;

View file

@ -162,11 +162,22 @@ impl Bus {
unsafe { unsafe {
let bus = self.to_glib_none().0; let bus = self.to_glib_none().0;
if !gobject_sys::g_object_get_qdata(bus as *mut _, SET_ONCE_QUARK.to_glib()).is_null() {
// This is not thread-safe before 1.16.3, see
// https://gitlab.freedesktop.org/gstreamer/gstreamer-rs/merge_requests/416
if ::version() < (1, 16, 3, 0) {
if !gobject_sys::g_object_get_qdata(bus as *mut _, SET_ONCE_QUARK.to_glib())
.is_null()
{
panic!("Bus sync handler can only be set once"); panic!("Bus sync handler can only be set once");
} }
gobject_sys::g_object_set_qdata(bus as *mut _, SET_ONCE_QUARK.to_glib(), 1 as *mut _); gobject_sys::g_object_set_qdata(
bus as *mut _,
SET_ONCE_QUARK.to_glib(),
1 as *mut _,
);
}
gst_sys::gst_bus_set_sync_handler( gst_sys::gst_bus_set_sync_handler(
bus, bus,
@ -177,6 +188,20 @@ impl Bus {
} }
} }
pub fn unset_sync_handler(&self) {
// This is not thread-safe before 1.16.3, see
// https://gitlab.freedesktop.org/gstreamer/gstreamer-rs/merge_requests/416
if ::version() < (1, 16, 3, 0) {
return;
}
unsafe {
use std::ptr;
gst_sys::gst_bus_set_sync_handler(self.to_glib_none().0, None, ptr::null_mut(), None)
}
}
pub fn iter(&self) -> Iter { pub fn iter(&self) -> Iter {
self.iter_timed(0.into()) self.iter_timed(0.into())
} }
@ -276,6 +301,12 @@ impl BusStream {
} }
} }
impl Drop for BusStream {
fn drop(&mut self) {
self.bus.unset_sync_handler();
}
}
impl Stream for BusStream { impl Stream for BusStream {
type Item = Message; type Item = Message;