summaryrefslogtreecommitdiffstats
path: root/third_party/rust/dbus/src/connection2.rs
diff options
context:
space:
mode:
Diffstat (limited to 'third_party/rust/dbus/src/connection2.rs')
-rw-r--r--third_party/rust/dbus/src/connection2.rs214
1 files changed, 214 insertions, 0 deletions
diff --git a/third_party/rust/dbus/src/connection2.rs b/third_party/rust/dbus/src/connection2.rs
new file mode 100644
index 0000000000..e318e921ee
--- /dev/null
+++ b/third_party/rust/dbus/src/connection2.rs
@@ -0,0 +1,214 @@
+use crate::{BusType, Error, Message, to_c_str, Watch};
+use std::{ptr, str};
+use std::ffi::CStr;
+use std::os::raw::{c_void};
+
+#[derive(Debug)]
+pub struct ConnHandle(*mut ffi::DBusConnection);
+
+unsafe impl Send for ConnHandle {}
+unsafe impl Sync for ConnHandle {}
+
+impl Drop for ConnHandle {
+ fn drop(&mut self) {
+ unsafe {
+ ffi::dbus_connection_close(self.0);
+ ffi::dbus_connection_unref(self.0);
+ }
+ }
+}
+
+/// Experimental rewrite of Connection [unstable / experimental]
+///
+/// Slightly lower level, with better support for async operations.
+/// Also, this struct is Send + Sync.
+///
+/// Blocking operations should be clearly marked as such, although if you
+/// try to access the connection from several threads at the same time,
+/// blocking might occur due to an internal mutex inside the dbus library.
+///
+/// This version avoids dbus_connection_dispatch, and thus avoids
+/// callbacks from that function. Instead the same functionality needs to be
+/// implemented by these bindings somehow - this is not done yet.
+#[derive(Debug)]
+pub struct TxRx {
+ handle: ConnHandle,
+}
+
+impl TxRx {
+ #[inline(always)]
+ pub (crate) fn conn(&self) -> *mut ffi::DBusConnection {
+ self.handle.0
+ }
+
+ fn conn_from_ptr(ptr: *mut ffi::DBusConnection) -> Result<TxRx, Error> {
+ let handle = ConnHandle(ptr);
+
+ /* No, we don't want our app to suddenly quit if dbus goes down */
+ unsafe { ffi::dbus_connection_set_exit_on_disconnect(ptr, 0) };
+
+ let c = TxRx { handle };
+
+ Ok(c)
+ }
+
+
+ /// Creates a new D-Bus connection.
+ ///
+ /// Blocking: until the connection is up and running.
+ pub fn get_private(bus: BusType) -> Result<TxRx, Error> {
+ let mut e = Error::empty();
+ let conn = unsafe { ffi::dbus_bus_get_private(bus, e.get_mut()) };
+ if conn == ptr::null_mut() {
+ return Err(e)
+ }
+ Self::conn_from_ptr(conn)
+ }
+
+ /// Creates a new D-Bus connection to a remote address.
+ ///
+ /// Note: for all common cases (System / Session bus) you probably want "get_private" instead.
+ ///
+ /// Blocking: until the connection is established.
+ pub fn open_private(address: &str) -> Result<TxRx, Error> {
+ let mut e = Error::empty();
+ let conn = unsafe { ffi::dbus_connection_open_private(to_c_str(address).as_ptr(), e.get_mut()) };
+ if conn == ptr::null_mut() {
+ return Err(e)
+ }
+ Self::conn_from_ptr(conn)
+ }
+
+ /// Registers a new D-Bus connection with the bus.
+ ///
+ /// Note: `get_private` does this automatically, useful with `open_private`
+ ///
+ /// Blocking: until a "Hello" response is received from the server.
+ pub fn register(&mut self) -> Result<(), Error> {
+ // This function needs to take &mut self, because it changes unique_name and unique_name takes a &self
+ let mut e = Error::empty();
+ if unsafe { ffi::dbus_bus_register(self.conn(), e.get_mut()) == 0 } {
+ Err(e)
+ } else {
+ Ok(())
+ }
+ }
+
+ /// Gets whether the connection is currently open.
+ pub fn is_connected(&self) -> bool {
+ unsafe { ffi::dbus_connection_get_is_connected(self.conn()) != 0 }
+ }
+
+ /// Get the connection's unique name.
+ ///
+ /// It's usually something like ":1.54"
+ pub fn unique_name(&self) -> Option<&str> {
+ let c = unsafe { ffi::dbus_bus_get_unique_name(self.conn()) };
+ if c == ptr::null_mut() { return None; }
+ let s = unsafe { CStr::from_ptr(c) };
+ str::from_utf8(s.to_bytes()).ok()
+ }
+
+
+ /// Puts a message into libdbus out queue. Use "flush" or "read_write" to make sure it is sent over the wire.
+ ///
+ /// Returns a serial number than can be used to match against a reply.
+ pub fn send(&self, msg: Message) -> Result<u32, ()> {
+ let mut serial = 0u32;
+ let r = unsafe { ffi::dbus_connection_send(self.conn(), msg.ptr(), &mut serial) };
+ if r == 0 { return Err(()); }
+ Ok(serial)
+ }
+
+ /// Flush the queue of outgoing messages.
+ ///
+ /// Blocking: until the outgoing queue is empty.
+ pub fn flush(&self) { unsafe { ffi::dbus_connection_flush(self.conn()) } }
+
+ /// Read and write to the connection.
+ ///
+ /// Incoming messages are put in the internal queue, outgoing messages are written.
+ ///
+ /// Blocking: If there are no messages, for up to timeout_ms milliseconds, or forever if timeout_ms is None.
+ /// For non-blocking behaviour, set timeout_ms to Some(0).
+ pub fn read_write(&self, timeout_ms: Option<i32>) -> Result<(), ()> {
+ let t = timeout_ms.unwrap_or(-1);
+ if unsafe { ffi::dbus_connection_read_write(self.conn(), t) == 0 } {
+ Err(())
+ } else {
+ Ok(())
+ }
+ }
+
+ /// Removes a message from the incoming queue, or returns None if the queue is empty.
+ ///
+ /// Use "read_write" first, so that messages are put into the incoming queue.
+ /// For unhandled messages, please call MessageDispatcher::default_dispatch to return
+ /// default replies for method calls.
+ pub fn pop_message(&self) -> Option<Message> {
+ let mptr = unsafe { ffi::dbus_connection_pop_message(self.conn()) };
+ if mptr == ptr::null_mut() {
+ None
+ } else {
+ Some(Message::from_ptr(mptr, false))
+ }
+ }
+
+ /// Get an up-to-date list of file descriptors to watch.
+ ///
+ /// Might be changed into something that allows for callbacks when the watch list is changed.
+ pub fn watch_fds(&mut self) -> Result<Vec<Watch>, ()> {
+ extern "C" fn add_watch_cb(watch: *mut ffi::DBusWatch, data: *mut c_void) -> u32 {
+ unsafe {
+ let wlist: &mut Vec<Watch> = &mut *(data as *mut _);
+ wlist.push(Watch::from_raw(watch));
+ }
+ 1
+ }
+ let mut r = vec!();
+ if unsafe { ffi::dbus_connection_set_watch_functions(self.conn(),
+ Some(add_watch_cb), None, None, &mut r as *mut _ as *mut _, None) } == 0 { return Err(()) }
+ assert!(unsafe { ffi::dbus_connection_set_watch_functions(self.conn(),
+ None, None, None, ptr::null_mut(), None) } != 0);
+ Ok(r)
+ }
+}
+
+#[test]
+fn test_txrx_send_sync() {
+ fn is_send<T: Send>(_: &T) {}
+ fn is_sync<T: Sync>(_: &T) {}
+ let c = TxRx::get_private(BusType::Session).unwrap();
+ is_send(&c);
+ is_sync(&c);
+}
+
+#[test]
+fn txrx_simple_test() {
+ let mut c = TxRx::get_private(BusType::Session).unwrap();
+ assert!(c.is_connected());
+ let fds = c.watch_fds().unwrap();
+ println!("{:?}", fds);
+ assert!(fds.len() > 0);
+ let m = Message::new_method_call("org.freedesktop.DBus", "/", "org.freedesktop.DBus", "ListNames").unwrap();
+ let reply = c.send(m).unwrap();
+ let my_name = c.unique_name().unwrap();
+ loop {
+ while let Some(mut msg) = c.pop_message() {
+ println!("{:?}", msg);
+ if msg.get_reply_serial() == Some(reply) {
+ let r = msg.as_result().unwrap();
+ let z: ::arg::Array<&str, _> = r.get1().unwrap();
+ for n in z {
+ println!("{}", n);
+ if n == my_name { return; } // Hooray, we found ourselves!
+ }
+ assert!(false);
+ } else if let Some(r) = crate::MessageDispatcher::<()>::default_dispatch(&msg) {
+ c.send(r).unwrap();
+ }
+ }
+ c.read_write(Some(100)).unwrap();
+ }
+}
+