mirror of
https://gitlab.freedesktop.org/gstreamer/gst-plugins-rs.git
synced 2025-01-10 11:15:33 +00:00
6596b6cdd1
It attempts to produce a (nearly) gapless live stream by synchronizing its output to the running time and forwarding the next input buffer if its start is (nearly) flush with the end of the last output buffer. If the input buffer is missing or too far in the future, it duplicates the last output buffer with adjusted timestamps. If it is operating on a raw audio stream, it will fill duplicate buffers with silence. If an input buffer arrives too late, it is thrown away. If the last input buffer was accepted too long ago (according to `late-threshold`), a late input buffer is accepted anyway, but immediately considered a duplicate. Due to the silence-filling, this has no effect on audio, but video gets a "slideshow" effect instead of freezing completely. The "many-repeats" property will be notified when this element has recently duplicated a lot of buffers or recovered from such a state. Co-authored-by: Vivia Nikolaidou <vivia@ahiru.eu> Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-rs/-/merge_requests/1017>
175 lines
5.2 KiB
Rust
175 lines
5.2 KiB
Rust
// Copyright (C) 2022 LTN Global Communications, Inc.
|
|
// Contact: Jan Alexander Steffens (heftig) <jan.steffens@ltnglobal.com>
|
|
//
|
|
// This Source Code Form is subject to the terms of the Mozilla Public License, v2.0.
|
|
// If a copy of the MPL was not distributed with this file, You can obtain one at
|
|
// <https://mozilla.org/MPL/2.0/>.
|
|
//
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
use gio::prelude::*;
|
|
use gst::{glib, prelude::*};
|
|
use gtk::prelude::*;
|
|
use std::cell::Cell;
|
|
|
|
struct DroppingProbe(glib::WeakRef<gst::Pad>, Option<gst::PadProbeId>);
|
|
|
|
impl DroppingProbe {
|
|
fn install(pad: &gst::Pad) -> Self {
|
|
let probe_id = pad
|
|
.add_probe(gst::PadProbeType::BUFFER, |_, _| gst::PadProbeReturn::Drop)
|
|
.unwrap();
|
|
Self(pad.downgrade(), Some(probe_id))
|
|
}
|
|
}
|
|
|
|
impl Drop for DroppingProbe {
|
|
fn drop(&mut self) {
|
|
if let Some((pad, probe_id)) = self.0.upgrade().zip(self.1.take()) {
|
|
pad.remove_probe(probe_id);
|
|
}
|
|
}
|
|
}
|
|
|
|
fn create_pipeline() -> gst::Pipeline {
|
|
gst::parse_launch(
|
|
r#"videotestsrc name=vsrc is-live=1
|
|
! video/x-raw,framerate=60/1,width=800,height=600
|
|
! timeoverlay text="Pre:"
|
|
! queue
|
|
! livesync latency=50000000
|
|
! videorate
|
|
! timeoverlay text="Post:" halignment=right
|
|
! queue
|
|
! gtk4paintablesink name=vsink
|
|
audiotestsrc name=asrc is-live=1
|
|
! audio/x-raw,channels=2
|
|
! queue
|
|
! livesync latency=50000000
|
|
! audiorate
|
|
! queue
|
|
! autoaudiosink
|
|
"#,
|
|
)
|
|
.expect("Failed to create pipeline")
|
|
.downcast()
|
|
.unwrap()
|
|
}
|
|
|
|
fn create_window(app: >k::Application) {
|
|
let pipeline = create_pipeline();
|
|
let video_src_pad = pipeline.by_name("vsrc").unwrap().static_pad("src").unwrap();
|
|
let audio_src_pad = pipeline.by_name("asrc").unwrap().static_pad("src").unwrap();
|
|
|
|
let window = gtk::ApplicationWindow::new(app);
|
|
window.set_default_size(800, 684);
|
|
|
|
let vbox = gtk::Box::new(gtk::Orientation::Vertical, 0);
|
|
|
|
let picture = gtk::Picture::new();
|
|
let paintable = pipeline
|
|
.by_name("vsink")
|
|
.unwrap()
|
|
.property::<gtk::gdk::Paintable>("paintable");
|
|
picture.set_paintable(Some(&paintable));
|
|
vbox.append(&picture);
|
|
|
|
let action_bar = gtk::ActionBar::new();
|
|
vbox.append(&action_bar);
|
|
|
|
let offset_spin = gtk::SpinButton::with_range(0.0, 500.0, 100.0);
|
|
action_bar.pack_start(&offset_spin);
|
|
|
|
{
|
|
let video_src_pad = video_src_pad.clone();
|
|
let audio_src_pad = audio_src_pad.clone();
|
|
offset_spin.connect_value_notify(move |offset_spin| {
|
|
const MSECOND: f64 = gst::ClockTime::MSECOND.nseconds() as _;
|
|
|
|
let offset = (offset_spin.value() * -MSECOND) as i64;
|
|
video_src_pad.set_offset(offset);
|
|
audio_src_pad.set_offset(offset);
|
|
});
|
|
}
|
|
|
|
let drop_button = gtk::ToggleButton::with_label("Drop Signal");
|
|
action_bar.pack_end(&drop_button);
|
|
|
|
let drop_ids = Cell::new(None);
|
|
drop_button.connect_toggled(move |drop_button| {
|
|
if drop_button.is_active() {
|
|
let video_probe = DroppingProbe::install(&video_src_pad);
|
|
let audio_probe = DroppingProbe::install(&audio_src_pad);
|
|
drop_ids.set(Some((video_probe, audio_probe)));
|
|
} else {
|
|
drop_ids.set(None);
|
|
}
|
|
});
|
|
|
|
{
|
|
let bus = pipeline.bus().unwrap();
|
|
let window = window.downgrade();
|
|
bus.add_watch_local(move |_, msg| {
|
|
use gst::MessageView;
|
|
|
|
match msg.view() {
|
|
MessageView::Eos(..) => {
|
|
if let Some(window) = window.upgrade() {
|
|
window.close();
|
|
}
|
|
}
|
|
|
|
MessageView::Error(err) => {
|
|
eprintln!(
|
|
"Error from {}: {} ({:?})",
|
|
msg.src().map(|s| s.path_string()).as_deref().unwrap_or(""),
|
|
err.error(),
|
|
err.debug(),
|
|
);
|
|
if let Some(window) = window.upgrade() {
|
|
window.close();
|
|
}
|
|
}
|
|
|
|
_ => (),
|
|
};
|
|
|
|
glib::Continue(true)
|
|
})
|
|
.unwrap();
|
|
}
|
|
|
|
{
|
|
let pipeline = pipeline.clone();
|
|
window.connect_realize(move |_| {
|
|
pipeline
|
|
.set_state(gst::State::Playing)
|
|
.expect("Failed to start pipeline");
|
|
});
|
|
}
|
|
|
|
window.connect_unrealize(move |_| {
|
|
pipeline
|
|
.set_state(gst::State::Null)
|
|
.expect("Failed to stop pipeline");
|
|
});
|
|
|
|
window.set_child(Some(&vbox));
|
|
window.show();
|
|
}
|
|
|
|
fn main() {
|
|
let app = gtk::Application::new(
|
|
Some("gtk-plugins-rs.gtk-livesync"),
|
|
gio::ApplicationFlags::FLAGS_NONE,
|
|
);
|
|
|
|
app.connect_startup(move |_app| {
|
|
gst::init().expect("Failed to initialize GStreamer");
|
|
gstlivesync::plugin_register_static().expect("Failed to register livesync plugin");
|
|
gstgtk4::plugin_register_static().expect("Failed to register gstgtk4 plugin");
|
|
});
|
|
|
|
app.connect_activate(create_window);
|
|
app.run();
|
|
}
|