forked from mirrors/gstreamer-rs
examples: appsrc: improved error handling
The program should now exit gracefully with human readable messages.
This commit is contained in:
parent
d6c032be7b
commit
50e6e2e108
1 changed files with 81 additions and 24 deletions
|
@ -3,24 +3,67 @@ use gst::*;
|
||||||
extern crate gstreamer_app as gst_app;
|
extern crate gstreamer_app as gst_app;
|
||||||
use gst_app::*;
|
use gst_app::*;
|
||||||
|
|
||||||
|
extern crate glib;
|
||||||
|
|
||||||
|
use std::fmt;
|
||||||
use std::u64;
|
use std::u64;
|
||||||
use std::thread;
|
use std::thread;
|
||||||
|
|
||||||
const WIDTH: usize = 320;
|
const WIDTH: usize = 320;
|
||||||
const HEIGHT: usize = 240;
|
const HEIGHT: usize = 240;
|
||||||
|
|
||||||
fn main() {
|
#[derive(Debug)]
|
||||||
gst::init().unwrap();
|
enum AppSrcExError {
|
||||||
|
InitFailed(glib::Error),
|
||||||
|
ElementNotFound(&'static str),
|
||||||
|
ElementLinkFailed(&'static str, &'static str),
|
||||||
|
SetStateError(&'static str),
|
||||||
|
ElementError(std::string::String, glib::Error, std::string::String),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for AppSrcExError {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
match *self {
|
||||||
|
AppSrcExError::InitFailed(ref e) => {
|
||||||
|
write!(f, "GStreamer initialization failed: {:?}", e)
|
||||||
|
}
|
||||||
|
AppSrcExError::ElementNotFound(ref e) => write!(f, "Element {} not found", e),
|
||||||
|
AppSrcExError::ElementLinkFailed(ref e1, ref e2) => {
|
||||||
|
write!(f, "Link failed between {} and {}", e1, e2)
|
||||||
|
}
|
||||||
|
AppSrcExError::SetStateError(ref state) => {
|
||||||
|
write!(f, "Pipeline failed to switch to state {}", state)
|
||||||
|
}
|
||||||
|
AppSrcExError::ElementError(ref element, ref err, ref debug) => {
|
||||||
|
write!(f, "Error from {}: {} ({:?})", element, err, debug)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn create_pipeline() -> Result<(Pipeline, AppSrc), AppSrcExError> {
|
||||||
|
gst::init().map_err(|e| AppSrcExError::InitFailed(e))?;
|
||||||
|
|
||||||
let pipeline = gst::Pipeline::new(None);
|
let pipeline = gst::Pipeline::new(None);
|
||||||
let src = gst::ElementFactory::make("appsrc", None).unwrap();
|
let src = gst::ElementFactory::make("appsrc", None)
|
||||||
let videoconvert = gst::ElementFactory::make("videoconvert", None).unwrap();
|
.ok_or(AppSrcExError::ElementNotFound("appsrc"))?;
|
||||||
let sink = gst::ElementFactory::make("autovideosink", None).unwrap();
|
|
||||||
|
|
||||||
pipeline.add_many(&[&src, &videoconvert, &sink]).unwrap();
|
let videoconvert = gst::ElementFactory::make("videoconvert", None)
|
||||||
gst::Element::link_many(&[&src, &videoconvert, &sink]).unwrap();
|
.ok_or(AppSrcExError::ElementNotFound("videoconvert"))?;
|
||||||
|
let sink = gst::ElementFactory::make("autovideosink", None)
|
||||||
|
.ok_or(AppSrcExError::ElementNotFound("autovideosink"))?;
|
||||||
|
|
||||||
let appsrc = src.clone().dynamic_cast::<AppSrc>().unwrap();
|
pipeline
|
||||||
|
.add_many(&[&src, &videoconvert, &sink])
|
||||||
|
.expect("Unable to add elements in the pipeline");
|
||||||
|
gst::Element::link(&src, &videoconvert)
|
||||||
|
.map_err(|_| AppSrcExError::ElementLinkFailed("src", "videoconvert"))?;
|
||||||
|
gst::Element::link(&videoconvert, &sink)
|
||||||
|
.map_err(|_| AppSrcExError::ElementLinkFailed("videoconvert", "sink"))?;
|
||||||
|
|
||||||
|
let appsrc = src.clone()
|
||||||
|
.dynamic_cast::<AppSrc>()
|
||||||
|
.expect("Source element is expected to be an appsrc!");
|
||||||
appsrc.set_caps(&Caps::new_simple(
|
appsrc.set_caps(&Caps::new_simple(
|
||||||
"video/x-raw",
|
"video/x-raw",
|
||||||
&[
|
&[
|
||||||
|
@ -34,6 +77,12 @@ fn main() {
|
||||||
appsrc.set_max_bytes(1);
|
appsrc.set_max_bytes(1);
|
||||||
appsrc.set_property_block(true);
|
appsrc.set_property_block(true);
|
||||||
|
|
||||||
|
Ok((pipeline, appsrc))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main_loop() -> Result<(), AppSrcExError> {
|
||||||
|
let (pipeline, appsrc) = create_pipeline()?;
|
||||||
|
|
||||||
thread::spawn(move || {
|
thread::spawn(move || {
|
||||||
for i in 0..100 {
|
for i in 0..100 {
|
||||||
println!("Producing frame {}", i);
|
println!("Producing frame {}", i);
|
||||||
|
@ -44,14 +93,14 @@ fn main() {
|
||||||
let g = if i % 3 == 0 { 0 } else { 255 };
|
let g = if i % 3 == 0 { 0 } else { 255 };
|
||||||
let b = if i % 5 == 0 { 0 } else { 255 };
|
let b = if i % 5 == 0 { 0 } else { 255 };
|
||||||
|
|
||||||
for _ in 0..(320 * 240) {
|
for _ in 0..(WIDTH * HEIGHT) {
|
||||||
vec.push(b);
|
vec.push(b);
|
||||||
vec.push(g);
|
vec.push(g);
|
||||||
vec.push(r);
|
vec.push(r);
|
||||||
vec.push(0);
|
vec.push(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut buffer = Buffer::from_vec(vec).unwrap();
|
let mut buffer = Buffer::from_vec(vec).expect("Unable to create a Buffer");
|
||||||
buffer.get_mut().unwrap().set_pts(i * 500_000_000);
|
buffer.get_mut().unwrap().set_pts(i * 500_000_000);
|
||||||
|
|
||||||
if appsrc.push_buffer(buffer) != FlowReturn::Ok {
|
if appsrc.push_buffer(buffer) != FlowReturn::Ok {
|
||||||
|
@ -62,12 +111,13 @@ fn main() {
|
||||||
appsrc.end_of_stream();
|
appsrc.end_of_stream();
|
||||||
});
|
});
|
||||||
|
|
||||||
assert_ne!(
|
if let gst::StateChangeReturn::Failure = pipeline.set_state(gst::State::Playing) {
|
||||||
pipeline.set_state(gst::State::Playing),
|
return Err(AppSrcExError::SetStateError("playing"));
|
||||||
gst::StateChangeReturn::Failure
|
}
|
||||||
);
|
|
||||||
|
|
||||||
let bus = pipeline.get_bus().unwrap();
|
let bus = pipeline
|
||||||
|
.get_bus()
|
||||||
|
.expect("Pipeline without bus. Shouldn't happen!");
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
let msg = match bus.timed_pop(u64::MAX) {
|
let msg = match bus.timed_pop(u64::MAX) {
|
||||||
|
@ -78,20 +128,27 @@ fn main() {
|
||||||
match msg.view() {
|
match msg.view() {
|
||||||
MessageView::Eos(..) => break,
|
MessageView::Eos(..) => break,
|
||||||
MessageView::Error(err) => {
|
MessageView::Error(err) => {
|
||||||
println!(
|
pipeline.set_state(gst::State::Null);
|
||||||
"Error from {}: {} ({:?})",
|
return Err(AppSrcExError::ElementError(
|
||||||
msg.get_src().get_path_string(),
|
msg.get_src().get_path_string(),
|
||||||
err.get_error(),
|
err.get_error(),
|
||||||
err.get_debug()
|
err.get_debug().unwrap(),
|
||||||
);
|
));
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
_ => (),
|
_ => (),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
assert_ne!(
|
if let gst::StateChangeReturn::Failure = pipeline.set_state(gst::State::Null) {
|
||||||
pipeline.set_state(gst::State::Null),
|
return Err(AppSrcExError::SetStateError("null"));
|
||||||
gst::StateChangeReturn::Failure
|
}
|
||||||
);
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
match main_loop() {
|
||||||
|
Ok(r) => r,
|
||||||
|
Err(e) => eprintln!("Error! {}", e),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue