1 | // Take a look at the license at the top of the repository in the LICENSE file. |
2 | |
3 | use std::{ |
4 | ops::Deref, |
5 | pin::Pin, |
6 | ptr, |
7 | task::{Context, Poll}, |
8 | }; |
9 | |
10 | use glib::translate::*; |
11 | |
12 | use crate::{PromiseResult, Structure, StructureRef}; |
13 | |
14 | glib::wrapper! { |
15 | #[derive (Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] |
16 | #[doc (alias = "GstPromise" )] |
17 | pub struct Promise(Shared<ffi::GstPromise>); |
18 | |
19 | match fn { |
20 | ref => |ptr| ffi::gst_mini_object_ref(ptr as *mut _), |
21 | unref => |ptr| ffi::gst_mini_object_unref(ptr as *mut _), |
22 | type_ => || ffi::gst_promise_get_type(), |
23 | } |
24 | } |
25 | |
26 | #[derive (Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)] |
27 | pub enum PromiseError { |
28 | Interrupted, |
29 | Expired, |
30 | Other(PromiseResult), |
31 | } |
32 | |
33 | impl Promise { |
34 | #[doc (alias = "gst_promise_new" )] |
35 | pub fn new() -> Promise { |
36 | assert_initialized_main_thread!(); |
37 | unsafe { from_glib_full(ffi::gst_promise_new()) } |
38 | } |
39 | |
40 | #[doc (alias = "gst_promise_new_with_change_func" )] |
41 | pub fn with_change_func<F>(func: F) -> Promise |
42 | where |
43 | F: FnOnce(Result<Option<&StructureRef>, PromiseError>) + Send + 'static, |
44 | { |
45 | assert_initialized_main_thread!(); |
46 | let user_data: Box<Option<F>> = Box::new(Some(func)); |
47 | |
48 | unsafe extern "C" fn trampoline< |
49 | F: FnOnce(Result<Option<&StructureRef>, PromiseError>) + Send + 'static, |
50 | >( |
51 | promise: *mut ffi::GstPromise, |
52 | user_data: glib::ffi::gpointer, |
53 | ) { |
54 | let user_data: &mut Option<F> = &mut *(user_data as *mut _); |
55 | let callback = user_data.take().unwrap(); |
56 | |
57 | let promise: Borrowed<Promise> = from_glib_borrow(promise); |
58 | |
59 | let res = match promise.wait() { |
60 | PromiseResult::Replied => Ok(promise.get_reply()), |
61 | PromiseResult::Interrupted => Err(PromiseError::Interrupted), |
62 | PromiseResult::Expired => Err(PromiseError::Expired), |
63 | PromiseResult::Pending => { |
64 | panic!("Promise resolved but returned Pending" ); |
65 | } |
66 | err => Err(PromiseError::Other(err)), |
67 | }; |
68 | |
69 | callback(res); |
70 | } |
71 | |
72 | unsafe extern "C" fn free_user_data< |
73 | F: FnOnce(Result<Option<&StructureRef>, PromiseError>) + Send + 'static, |
74 | >( |
75 | user_data: glib::ffi::gpointer, |
76 | ) { |
77 | let _: Box<Option<F>> = Box::from_raw(user_data as *mut _); |
78 | } |
79 | |
80 | unsafe { |
81 | from_glib_full(ffi::gst_promise_new_with_change_func( |
82 | Some(trampoline::<F>), |
83 | Box::into_raw(user_data) as *mut _, |
84 | Some(free_user_data::<F>), |
85 | )) |
86 | } |
87 | } |
88 | |
89 | pub fn new_future() -> (Self, PromiseFuture) { |
90 | use futures_channel::oneshot; |
91 | |
92 | // We only use the channel as a convenient waker |
93 | let (sender, receiver) = oneshot::channel(); |
94 | let promise = Self::with_change_func(move |_res| { |
95 | let _ = sender.send(()); |
96 | }); |
97 | |
98 | (promise.clone(), PromiseFuture(promise, receiver)) |
99 | } |
100 | |
101 | #[doc (alias = "gst_promise_expire" )] |
102 | pub fn expire(&self) { |
103 | unsafe { |
104 | ffi::gst_promise_expire(self.to_glib_none().0); |
105 | } |
106 | } |
107 | |
108 | #[doc (alias = "gst_promise_get_reply" )] |
109 | pub fn get_reply(&self) -> Option<&StructureRef> { |
110 | unsafe { |
111 | let s = ffi::gst_promise_get_reply(self.to_glib_none().0); |
112 | if s.is_null() { |
113 | None |
114 | } else { |
115 | Some(StructureRef::from_glib_borrow(s)) |
116 | } |
117 | } |
118 | } |
119 | |
120 | #[doc (alias = "gst_promise_interrupt" )] |
121 | pub fn interrupt(&self) { |
122 | unsafe { |
123 | ffi::gst_promise_interrupt(self.to_glib_none().0); |
124 | } |
125 | } |
126 | |
127 | #[doc (alias = "gst_promise_reply" )] |
128 | pub fn reply(&self, s: Option<Structure>) { |
129 | unsafe { |
130 | ffi::gst_promise_reply( |
131 | self.to_glib_none().0, |
132 | s.map(|s| s.into_glib_ptr()).unwrap_or(ptr::null_mut()), |
133 | ); |
134 | } |
135 | } |
136 | |
137 | #[doc (alias = "gst_promise_wait" )] |
138 | pub fn wait(&self) -> PromiseResult { |
139 | unsafe { from_glib(ffi::gst_promise_wait(self.to_glib_none().0)) } |
140 | } |
141 | } |
142 | |
143 | impl Default for Promise { |
144 | fn default() -> Self { |
145 | Self::new() |
146 | } |
147 | } |
148 | |
149 | unsafe impl Send for Promise {} |
150 | unsafe impl Sync for Promise {} |
151 | |
152 | #[derive (Debug)] |
153 | pub struct PromiseFuture(Promise, futures_channel::oneshot::Receiver<()>); |
154 | |
155 | pub struct PromiseReply(Promise); |
156 | |
157 | impl std::future::Future for PromiseFuture { |
158 | type Output = Result<Option<PromiseReply>, PromiseError>; |
159 | |
160 | fn poll(mut self: Pin<&mut Self>, context: &mut Context) -> Poll<Self::Output> { |
161 | match Pin::new(&mut self.1).poll(context) { |
162 | Poll::Ready(Err(_)) => panic!("Sender dropped before callback was called" ), |
163 | Poll::Ready(Ok(())) => { |
164 | let res = match self.0.wait() { |
165 | PromiseResult::Replied => { |
166 | if self.0.get_reply().is_none() { |
167 | Ok(None) |
168 | } else { |
169 | Ok(Some(PromiseReply(self.0.clone()))) |
170 | } |
171 | } |
172 | PromiseResult::Interrupted => Err(PromiseError::Interrupted), |
173 | PromiseResult::Expired => Err(PromiseError::Expired), |
174 | PromiseResult::Pending => { |
175 | panic!("Promise resolved but returned Pending" ); |
176 | } |
177 | err => Err(PromiseError::Other(err)), |
178 | }; |
179 | Poll::Ready(res) |
180 | } |
181 | Poll::Pending => Poll::Pending, |
182 | } |
183 | } |
184 | } |
185 | |
186 | impl futures_core::future::FusedFuture for PromiseFuture { |
187 | fn is_terminated(&self) -> bool { |
188 | self.1.is_terminated() |
189 | } |
190 | } |
191 | |
192 | impl Deref for PromiseReply { |
193 | type Target = StructureRef; |
194 | |
195 | #[inline ] |
196 | fn deref(&self) -> &StructureRef { |
197 | self.0.get_reply().expect(msg:"Promise without reply" ) |
198 | } |
199 | } |
200 | |
201 | impl std::fmt::Debug for PromiseReply { |
202 | fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { |
203 | let mut debug: DebugTuple<'_, '_> = f.debug_tuple(name:"PromiseReply" ); |
204 | |
205 | match&mut DebugTuple<'_, '_> self.0.get_reply() { |
206 | Some(reply: &StructureRef) => debug.field(reply), |
207 | None => debug.field(&"<no reply>" ), |
208 | } |
209 | .finish() |
210 | } |
211 | } |
212 | |
213 | #[cfg (test)] |
214 | mod tests { |
215 | use std::{sync::mpsc::channel, thread}; |
216 | |
217 | use super::*; |
218 | |
219 | #[test ] |
220 | fn test_change_func() { |
221 | crate::init().unwrap(); |
222 | |
223 | let (sender, receiver) = channel(); |
224 | let promise = Promise::with_change_func(move |res| { |
225 | sender.send(res.map(|s| s.map(ToOwned::to_owned))).unwrap(); |
226 | }); |
227 | |
228 | thread::spawn(move || { |
229 | promise.reply(Some(crate::Structure::new_empty("foo/bar" ))); |
230 | }); |
231 | |
232 | let res = receiver.recv().unwrap(); |
233 | let res = res.expect("promise failed" ).expect("promise returned None" ); |
234 | assert_eq!(res.name(), "foo/bar" ); |
235 | } |
236 | } |
237 | |