mirror of
https://gitlab.freedesktop.org/gstreamer/gst-plugins-rs.git
synced 2024-11-22 03:21:00 +00:00
webrtcsink: expose consumer-pipeline-created signal
This signal is emitted as soon as the pipeline for each consumer is created, and can be used by applications that require a greater level of control over webrtcsink's internals. An example is also provided to demonstrate usage Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-plugins-rs/-/merge_requests/1220>
This commit is contained in:
parent
a27be7d054
commit
a20855dfd9
4 changed files with 163 additions and 4 deletions
|
@ -6466,6 +6466,20 @@
|
|||
"return-type": "void",
|
||||
"when": "last"
|
||||
},
|
||||
"consumer-pipeline-created": {
|
||||
"args": [
|
||||
{
|
||||
"name": "arg0",
|
||||
"type": "gchararray"
|
||||
},
|
||||
{
|
||||
"name": "arg1",
|
||||
"type": "GstPipeline"
|
||||
}
|
||||
],
|
||||
"return-type": "void",
|
||||
"when": "last"
|
||||
},
|
||||
"consumer-removed": {
|
||||
"args": [
|
||||
{
|
||||
|
|
|
@ -82,3 +82,6 @@ requires_private = "gstreamer-rtp-1.0 >= 1.20, gstreamer-webrtc-1.0 >= 1.20, gst
|
|||
|
||||
[[example]]
|
||||
name = "webrtcsink-stats-server"
|
||||
|
||||
[[example]]
|
||||
name = "webrtcsink-high-quality-tune"
|
||||
|
|
121
net/webrtc/examples/webrtcsink-high-quality-tune.rs
Normal file
121
net/webrtc/examples/webrtcsink-high-quality-tune.rs
Normal file
|
@ -0,0 +1,121 @@
|
|||
// The goal of this example is to demonstrate how to tune webrtcsink for a
|
||||
// high-quality, single consumer use case
|
||||
//
|
||||
// By default webrtcsink will use properties on elements such as videoscale
|
||||
// or the video encoders with the intent of maximising the potential number
|
||||
// of concurrent consumers while achieving a somewhat decent quality.
|
||||
//
|
||||
// In cases where the application knows that CPU usage will not be a concern,
|
||||
// for instance because there will only ever be a single concurrent consumer,
|
||||
// or it is running on a supercomputer, it may wish to maximize quality instead.
|
||||
//
|
||||
// This example can be used as a starting point by applications that need an
|
||||
// increased amount of control over webrtcsink internals, bearing in mind that
|
||||
// webrtcsink does not guarantee stability of said internals.
|
||||
|
||||
use anyhow::Error;
|
||||
use gst::prelude::*;
|
||||
|
||||
fn main() -> Result<(), Error> {
|
||||
gst::init()?;
|
||||
|
||||
// Create a very simple webrtc producer, offering a single video stream
|
||||
let pipeline = gst::Pipeline::builder().build();
|
||||
|
||||
let videotestsrc = gst::ElementFactory::make("videotestsrc").build()?;
|
||||
let queue = gst::ElementFactory::make("queue").build()?;
|
||||
let webrtcsink = gst::ElementFactory::make("webrtcsink").build()?;
|
||||
|
||||
// For the sake of the example we will force H264
|
||||
webrtcsink.set_property_from_str("video-caps", "video/x-h264");
|
||||
|
||||
// We want to tweak how webrtcsink performs video scaling when needed, as
|
||||
// this can have a very visible impact over quality.
|
||||
//
|
||||
// To achieve that, we will connect to deep-element-added on the consumer
|
||||
// pipeline.
|
||||
webrtcsink.connect("consumer-pipeline-created", false, |values| {
|
||||
let pipeline = values[2].get::<gst::Pipeline>().unwrap();
|
||||
|
||||
pipeline.connect("deep-element-added", false, |values| {
|
||||
let element = values[2].get::<gst::Element>().unwrap();
|
||||
|
||||
if let Some(factory) = element.factory() {
|
||||
if factory.name().as_str() == "videoscale" {
|
||||
println!("Tuning videoscale");
|
||||
element.set_property_from_str("method", "lanczos");
|
||||
}
|
||||
}
|
||||
|
||||
None
|
||||
});
|
||||
|
||||
None
|
||||
});
|
||||
|
||||
// We *could* access the consumer encoder from our
|
||||
// consumer-pipeline-created handler, but doing so from an encoder-setup
|
||||
// callback is better practice, as it will also get called
|
||||
// when running the discovery pipelines, and changing properties on the
|
||||
// encoder may in theory affect the caps it outputs.
|
||||
webrtcsink.connect("encoder-setup", true, |values| {
|
||||
let encoder = values[3].get::<gst::Element>().unwrap();
|
||||
|
||||
println!("Encoder: {}", encoder.factory().unwrap().name());
|
||||
|
||||
if let Some(factory) = encoder.factory() {
|
||||
match factory.name().as_str() {
|
||||
"x264enc" => {
|
||||
println!("Applying extra configuration to x264enc");
|
||||
encoder.set_property_from_str("speed-preset", "medium");
|
||||
}
|
||||
name => {
|
||||
println!(
|
||||
"Can't tune unsupported H264 encoder {name}, \
|
||||
set GST_PLUGIN_FEATURE_RANK=x264enc:1000 when \
|
||||
running the example"
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Some(false.to_value())
|
||||
});
|
||||
|
||||
pipeline.add_many([&videotestsrc, &queue, &webrtcsink])?;
|
||||
gst::Element::link_many([&videotestsrc, &queue, &webrtcsink])?;
|
||||
|
||||
// Now we simply run the pipeline to completion
|
||||
|
||||
pipeline.set_state(gst::State::Playing)?;
|
||||
|
||||
let bus = pipeline.bus().expect("Pipeline should have a bus");
|
||||
|
||||
for msg in bus.iter_timed(gst::ClockTime::NONE) {
|
||||
use gst::MessageView;
|
||||
|
||||
match msg.view() {
|
||||
MessageView::Eos(..) => {
|
||||
println!("EOS");
|
||||
break;
|
||||
}
|
||||
MessageView::Error(err) => {
|
||||
pipeline.set_state(gst::State::Null)?;
|
||||
eprintln!(
|
||||
"Got error from {}: {} ({})",
|
||||
msg.src()
|
||||
.map(|s| String::from(s.path_string()))
|
||||
.unwrap_or_else(|| "None".into()),
|
||||
err.error(),
|
||||
err.debug().unwrap_or_else(|| "".into()),
|
||||
);
|
||||
break;
|
||||
}
|
||||
_ => (),
|
||||
}
|
||||
}
|
||||
|
||||
pipeline.set_state(gst::State::Null)?;
|
||||
|
||||
Ok(())
|
||||
}
|
|
@ -1941,6 +1941,13 @@ impl BaseWebRTCSink {
|
|||
peer_id: &str,
|
||||
offer: Option<&gst_webrtc::WebRTCSessionDescription>,
|
||||
) -> Result<(), WebRTCSinkError> {
|
||||
let pipeline = gst::Pipeline::builder()
|
||||
.name(format!("session-pipeline-{session_id}"))
|
||||
.build();
|
||||
|
||||
self.obj()
|
||||
.emit_by_name::<()>("consumer-pipeline-created", &[&peer_id, &pipeline]);
|
||||
|
||||
let settings = self.settings.lock().unwrap();
|
||||
let mut state = self.state.lock().unwrap();
|
||||
let peer_id = peer_id.to_string();
|
||||
|
@ -1959,10 +1966,6 @@ impl BaseWebRTCSink {
|
|||
session_id
|
||||
);
|
||||
|
||||
let pipeline = gst::Pipeline::builder()
|
||||
.name(format!("session-pipeline-{session_id}"))
|
||||
.build();
|
||||
|
||||
let webrtcbin = make_element("webrtcbin", Some(&format!("webrtcbin-{session_id}")))
|
||||
.map_err(|err| WebRTCSinkError::SessionPipelineError {
|
||||
session_id: session_id.clone(),
|
||||
|
@ -3340,6 +3343,24 @@ impl ObjectImpl for BaseWebRTCSink {
|
|||
glib::subclass::Signal::builder("consumer-added")
|
||||
.param_types([String::static_type(), gst::Element::static_type()])
|
||||
.build(),
|
||||
/**
|
||||
* RsBaseWebRTCSink::consumer-pipeline-created:
|
||||
* @consumer_id: Identifier of the consumer
|
||||
* @pipeline: The pipeline that was just created
|
||||
*
|
||||
* This signal is emitted right after the pipeline for a new consumer
|
||||
* has been created, for instance allowing handlers to connect to
|
||||
* #GstBin::deep-element-added and tweak properties of any element used
|
||||
* by the pipeline.
|
||||
*
|
||||
* This provides access to the lower level components of webrtcsink, and
|
||||
* no guarantee is made that its internals will remain stable, use with caution!
|
||||
*
|
||||
* This is emitted *before* #RsBaseWebRTCSink::consumer-added .
|
||||
*/
|
||||
glib::subclass::Signal::builder("consumer-pipeline-created")
|
||||
.param_types([String::static_type(), gst::Pipeline::static_type()])
|
||||
.build(),
|
||||
/**
|
||||
* RsBaseWebRTCSink::consumer_removed:
|
||||
* @consumer_id: Identifier of the consumer that was removed
|
||||
|
|
Loading…
Reference in a new issue