gstreamer/element: Add call_async_future() that returns a future

The future would resolve into the return value of the closure that is
called asynchronously on the thread pool, and allows asynchronous
awaiting for it to finish.

  let res = element.call_async(|element| {
      element.set_state(gst::State::Playing)
  }).await;

  assert_eq!(res, Ok(gst::StateChangeSuccess::Success))
This commit is contained in:
Sebastian Dröge 2020-01-20 11:59:13 +02:00
parent c29a7638d3
commit db1c341cdf
3 changed files with 40 additions and 0 deletions

View file

@ -23,6 +23,8 @@ glib = { version = "0.9" }
num-rational = { version = "0.2", default-features = false, features = [] } num-rational = { version = "0.2", default-features = false, features = [] }
lazy_static = "1.0" lazy_static = "1.0"
futures-core = "0.3" futures-core = "0.3"
futures-util = "0.3"
futures-channel = "0.3"
muldiv = "0.2" muldiv = "0.2"
serde = { version = "1.0", optional = true } serde = { version = "1.0", optional = true }
serde_bytes = { version = "0.11", optional = true } serde_bytes = { version = "0.11", optional = true }

View file

@ -38,7 +38,13 @@ use StateChangeReturn;
use StateChangeSuccess; use StateChangeSuccess;
use std::ffi::CStr; use std::ffi::CStr;
#[cfg(any(feature = "v1_10", feature = "dox"))]
use std::future::Future;
#[cfg(any(feature = "v1_10", feature = "dox"))]
use std::marker::Unpin;
use std::mem; use std::mem;
#[cfg(any(feature = "v1_10", feature = "dox"))]
use std::pin::Pin;
use libc; use libc;
@ -247,6 +253,15 @@ pub trait ElementExtManual: 'static {
fn call_async<F>(&self, func: F) fn call_async<F>(&self, func: F)
where where
F: FnOnce(&Self) + Send + 'static; F: FnOnce(&Self) + Send + 'static;
#[cfg(any(feature = "v1_10", feature = "dox"))]
fn call_async_future<F, T>(
&self,
func: F,
) -> Pin<Box<dyn Future<Output = T> + Unpin + Send + 'static>>
where
F: FnOnce(&Self) -> T + Send + 'static,
T: Send + 'static;
} }
impl<O: IsA<Element>> ElementExtManual for O { impl<O: IsA<Element>> ElementExtManual for O {
@ -768,6 +783,27 @@ impl<O: IsA<Element>> ElementExtManual for O {
); );
} }
} }
#[cfg(any(feature = "v1_10", feature = "dox"))]
fn call_async_future<F, T>(
&self,
func: F,
) -> Pin<Box<dyn Future<Output = T> + Unpin + Send + 'static>>
where
F: FnOnce(&Self) -> T + Send + 'static,
T: Send + 'static,
{
use futures_channel::oneshot;
use futures_util::future::FutureExt;
let (sender, receiver) = oneshot::channel();
self.call_async(move |element| {
let _ = sender.send(func(element));
});
Box::pin(receiver.map(|res| res.expect("sender dropped")))
}
} }
impl ElementClass { impl ElementClass {

View file

@ -32,7 +32,9 @@ pub extern crate glib;
extern crate num_rational; extern crate num_rational;
extern crate futures_channel;
extern crate futures_core; extern crate futures_core;
extern crate futures_util;
extern crate muldiv; extern crate muldiv;