gstreamer-rs/examples/src/bin/appsink.rs

174 lines
4.8 KiB
Rust
Raw Normal View History

#[macro_use]
2017-08-01 18:44:01 +00:00
extern crate gstreamer as gst;
use gst::prelude::*;
2017-08-01 18:44:01 +00:00
extern crate gstreamer_app as gst_app;
extern crate gstreamer_audio as gst_audio;
2017-08-01 18:44:01 +00:00
extern crate glib;
extern crate byte_slice_cast;
use byte_slice_cast::*;
2018-04-01 08:30:03 +00:00
use std::error::Error as StdError;
2017-08-01 18:44:01 +00:00
use std::i16;
use std::i32;
extern crate failure;
use failure::Error;
#[macro_use]
extern crate failure_derive;
2017-11-12 18:07:02 +00:00
#[path = "../examples-common.rs"]
mod examples_common;
#[derive(Debug, Fail)]
#[fail(display = "Missing element {}", _0)]
struct MissingElement(&'static str);
#[derive(Debug, Fail)]
2018-07-27 10:36:40 +00:00
#[fail(
display = "Received error from {}: {} (debug: {:?})",
2018-10-28 13:47:02 +00:00
src, error, debug
2018-07-27 10:36:40 +00:00
)]
struct ErrorMessage {
src: String,
error: String,
debug: Option<String>,
#[cause]
cause: glib::Error,
}
2017-08-01 18:44:01 +00:00
fn create_pipeline() -> Result<gst::Pipeline, Error> {
gst::init()?;
let pipeline = gst::Pipeline::new(None);
let src =
gst::ElementFactory::make("audiotestsrc", None).ok_or(MissingElement("audiotestsrc"))?;
let sink = gst::ElementFactory::make("appsink", None).ok_or(MissingElement("appsink"))?;
pipeline.add_many(&[&src, &sink])?;
src.link(&sink)?;
let appsink = sink
.dynamic_cast::<gst_app::AppSink>()
.expect("Sink element is expected to be an appsink!");
2017-08-01 18:44:01 +00:00
appsink.set_caps(&gst::Caps::new_simple(
2017-08-01 18:44:01 +00:00
"audio/x-raw",
&[
("format", &gst_audio::AUDIO_FORMAT_S16.to_string()),
2017-08-02 17:15:16 +00:00
("layout", &"interleaved"),
("channels", &(1i32)),
("rate", &gst::IntRange::<i32>::new(1, i32::MAX)),
2017-08-01 18:44:01 +00:00
],
));
appsink.set_callbacks(
gst_app::AppSinkCallbacks::new()
.new_sample(|appsink| {
let sample = match appsink.pull_sample() {
None => return gst::FlowReturn::Eos,
Some(sample) => sample,
};
let buffer = if let Some(buffer) = sample.get_buffer() {
buffer
} else {
gst_element_error!(
appsink,
gst::ResourceError::Failed,
("Failed to get buffer from appsink")
);
return gst::FlowReturn::Error;
};
let map = if let Some(map) = buffer.map_readable() {
map
} else {
gst_element_error!(
appsink,
gst::ResourceError::Failed,
("Failed to map buffer readable")
);
return gst::FlowReturn::Error;
};
let samples = if let Ok(samples) = map.as_slice_of::<i16>() {
samples
} else {
gst_element_error!(
appsink,
gst::ResourceError::Failed,
("Failed to interprete buffer as S16 PCM")
);
return gst::FlowReturn::Error;
};
let sum: f64 = samples
.iter()
.map(|sample| {
let f = f64::from(*sample) / f64::from(i16::MAX);
f * f
2018-10-08 12:02:23 +00:00
})
.sum();
let rms = (sum / (samples.len() as f64)).sqrt();
println!("rms: {}", rms);
gst::FlowReturn::Ok
2018-10-08 12:02:23 +00:00
})
.build(),
);
2017-08-01 18:44:01 +00:00
Ok(pipeline)
}
2017-11-11 15:56:37 +00:00
fn main_loop(pipeline: gst::Pipeline) -> Result<(), Error> {
pipeline.set_state(gst::State::Playing).into_result()?;
let bus = pipeline
.get_bus()
.expect("Pipeline without bus. Shouldn't happen!");
2017-08-01 18:44:01 +00:00
while let Some(msg) = bus.timed_pop(gst::CLOCK_TIME_NONE) {
use gst::MessageView;
2017-08-01 18:44:01 +00:00
match msg.view() {
MessageView::Eos(..) => break,
MessageView::Error(err) => {
pipeline.set_state(gst::State::Null).into_result()?;
Err(ErrorMessage {
2018-07-27 10:36:40 +00:00
src: err
.get_src()
.map(|s| s.get_path_string())
.unwrap_or_else(|| String::from("None")),
error: err.get_error().description().into(),
debug: err.get_debug(),
cause: err.get_error(),
})?;
2017-08-01 18:44:01 +00:00
}
_ => (),
}
}
pipeline.set_state(gst::State::Null).into_result()?;
Ok(())
}
2017-11-12 18:07:02 +00:00
fn example_main() {
2017-11-11 15:56:37 +00:00
match create_pipeline().and_then(main_loop) {
Ok(r) => r,
Err(e) => eprintln!("Error! {}", e),
}
2017-08-01 18:44:01 +00:00
}
2017-11-12 18:07:02 +00:00
fn main() {
// tutorials_common::run is only required to set up the application environent on macOS
// (but not necessary in normal Cocoa applications where this is set up autmatically)
examples_common::run(example_main);
}