summaryrefslogtreecommitdiffstats
path: root/third_party/rust/dbus/src/arg/msgarg.rs
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 19:33:14 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 19:33:14 +0000
commit36d22d82aa202bb199967e9512281e9a53db42c9 (patch)
tree105e8c98ddea1c1e4784a60a5a6410fa416be2de /third_party/rust/dbus/src/arg/msgarg.rs
parentInitial commit. (diff)
downloadfirefox-esr-upstream.tar.xz
firefox-esr-upstream.zip
Adding upstream version 115.7.0esr.upstream/115.7.0esrupstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'third_party/rust/dbus/src/arg/msgarg.rs')
-rw-r--r--third_party/rust/dbus/src/arg/msgarg.rs426
1 files changed, 426 insertions, 0 deletions
diff --git a/third_party/rust/dbus/src/arg/msgarg.rs b/third_party/rust/dbus/src/arg/msgarg.rs
new file mode 100644
index 0000000000..3309c35e65
--- /dev/null
+++ b/third_party/rust/dbus/src/arg/msgarg.rs
@@ -0,0 +1,426 @@
+#![allow(dead_code)]
+
+use {Signature, Message, arg::TypeMismatchError};
+use std::{fmt, any};
+use std::sync::Arc;
+use std::rc::Rc;
+
+use super::{Iter, IterAppend, ArgType};
+
+/// Types that can represent a D-Bus message argument implement this trait.
+///
+/// Types should also implement either Append or Get to be useful.
+pub trait Arg {
+ /// The corresponding D-Bus argument type code.
+ const ARG_TYPE: ArgType;
+ /// The corresponding D-Bus argument type code; just returns ARG_TYPE.
+ ///
+ /// For backwards compatibility.
+ #[deprecated(note = "Use associated constant ARG_TYPE instead")]
+ fn arg_type() -> ArgType { return Self::ARG_TYPE; }
+ /// The corresponding D-Bus type signature for this type.
+ fn signature() -> Signature<'static>;
+}
+
+/// Types that can be appended to a message as arguments implement this trait.
+pub trait Append: Sized {
+ /// Performs the append operation.
+ fn append(self, &mut IterAppend);
+}
+
+/// Helper trait to append many arguments to a message.
+pub trait AppendAll: Sized {
+ /// Performs the append operation.
+ fn append(self, &mut IterAppend);
+}
+
+/// Types that can be retrieved from a message as arguments implement this trait.
+pub trait Get<'a>: Sized {
+ /// Performs the get operation.
+ fn get(i: &mut Iter<'a>) -> Option<Self>;
+}
+
+/// Helper trait to read all arguments from a message.
+pub trait ReadAll: Sized {
+ /// Performs the read operation.
+ fn read(i: &mut Iter) -> Result<Self, TypeMismatchError>;
+}
+
+
+/// Object safe version of Arg + Append + Get.
+pub trait RefArg: fmt::Debug {
+ /// The corresponding D-Bus argument type code.
+ fn arg_type(&self) -> ArgType;
+ /// The corresponding D-Bus type signature for this type.
+ fn signature(&self) -> Signature<'static>;
+ /// Performs the append operation.
+ fn append(&self, &mut IterAppend);
+ /// Transforms this argument to Any (which can be downcasted to read the current value).
+ ///
+ /// Note: The internal representation of complex types (Array, Dict, Struct) is unstable
+ /// and as_any should not be relied upon for these types. Use as_iter instead.
+ fn as_any(&self) -> &any::Any where Self: 'static;
+ /// Transforms this argument to Any (which can be downcasted to read the current value).
+ ///
+ /// Note: The internal representation of complex types (Array, Dict, Struct) is unstable
+ /// and as_any should not be relied upon for these types. Use as_iter instead.
+ ///
+ /// # Panic
+ /// Will panic if the interior cannot be made mutable, e g, if encapsulated
+ /// inside a Rc with a reference count > 1.
+ fn as_any_mut(&mut self) -> &mut any::Any where Self: 'static;
+ /// Try to read the argument as an i64.
+ ///
+ /// Works for: Boolean, Byte, Int16, UInt16, Int32, UInt32, Int64, UnixFd.
+ #[inline]
+ fn as_i64(&self) -> Option<i64> { None }
+ /// Try to read the argument as an u64.
+ ///
+ /// Works for: Boolean, Byte, Int16, UInt16, Int32, UInt32, UInt64.
+ #[inline]
+ fn as_u64(&self) -> Option<u64> { None }
+ /// Try to read the argument as an f64.
+ ///
+ /// Works for: Boolean, Byte, Int16, UInt16, Int32, UInt32, Double.
+ #[inline]
+ fn as_f64(&self) -> Option<f64> { None }
+ /// Try to read the argument as a str.
+ ///
+ /// Works for: String, ObjectPath, Signature.
+ #[inline]
+ fn as_str(&self) -> Option<&str> { None }
+ /// Try to read the argument as an iterator.
+ ///
+ /// Works for: Array/Dict, Struct, Variant.
+ #[inline]
+ fn as_iter<'a>(&'a self) -> Option<Box<Iterator<Item=&'a RefArg> + 'a>> { None }
+ /// Deep clone of the RefArg, causing the result to be 'static.
+ ///
+ /// Usable as an escape hatch in case of lifetime problems with RefArg.
+ ///
+ /// In case of complex types (Array, Dict, Struct), the clone is not guaranteed
+ /// to have the same internal representation as the original.
+ fn box_clone(&self) -> Box<RefArg + 'static> { unimplemented!() /* Needed for backwards comp */ }
+}
+
+impl<'a> Get<'a> for Box<RefArg> {
+ fn get(i: &mut Iter<'a>) -> Option<Self> { i.get_refarg() }
+}
+
+/// Cast a RefArg as a specific type (shortcut for any + downcast)
+#[inline]
+pub fn cast<'a, T: 'static>(a: &'a (RefArg + 'static)) -> Option<&'a T> { a.as_any().downcast_ref() }
+
+/// Cast a RefArg as a specific type (shortcut for any_mut + downcast_mut)
+///
+/// # Panic
+/// Will panic if the interior cannot be made mutable, e g, if encapsulated
+/// inside a Rc with a reference count > 1.
+#[inline]
+pub fn cast_mut<'a, T: 'static>(a: &'a mut (RefArg + 'static)) -> Option<&'a mut T> { a.as_any_mut().downcast_mut() }
+
+/// If a type implements this trait, it means the size and alignment is the same
+/// as in D-Bus. This means that you can quickly append and get slices of this type.
+///
+/// Note: Booleans do not implement this trait because D-Bus booleans are 4 bytes and Rust booleans are 1 byte.
+pub unsafe trait FixedArray: Arg + 'static + Clone + Copy {}
+
+/// Types that can be used as keys in a dict type implement this trait.
+pub trait DictKey: Arg {}
+
+
+
+/// Simple lift over reference to value - this makes some iterators more ergonomic to use
+impl<'a, T: Arg> Arg for &'a T {
+ const ARG_TYPE: ArgType = T::ARG_TYPE;
+ fn signature() -> Signature<'static> { T::signature() }
+}
+impl<'a, T: Append + Clone> Append for &'a T {
+ fn append(self, i: &mut IterAppend) { self.clone().append(i) }
+}
+impl<'a, T: DictKey> DictKey for &'a T {}
+
+impl<'a, T: RefArg + ?Sized> RefArg for &'a T {
+ #[inline]
+ fn arg_type(&self) -> ArgType { (&**self).arg_type() }
+ #[inline]
+ fn signature(&self) -> Signature<'static> { (&**self).signature() }
+ #[inline]
+ fn append(&self, i: &mut IterAppend) { (&**self).append(i) }
+ #[inline]
+ fn as_any(&self) -> &any::Any where T: 'static { (&**self).as_any() }
+ #[inline]
+ fn as_any_mut(&mut self) -> &mut any::Any where T: 'static { unreachable!() }
+ #[inline]
+ fn as_i64(&self) -> Option<i64> { (&**self).as_i64() }
+ #[inline]
+ fn as_u64(&self) -> Option<u64> { (&**self).as_u64() }
+ #[inline]
+ fn as_f64(&self) -> Option<f64> { (&**self).as_f64() }
+ #[inline]
+ fn as_str(&self) -> Option<&str> { (&**self).as_str() }
+ #[inline]
+ fn as_iter<'b>(&'b self) -> Option<Box<Iterator<Item=&'b RefArg> + 'b>> { (&**self).as_iter() }
+ #[inline]
+ fn box_clone(&self) -> Box<RefArg + 'static> { (&**self).box_clone() }
+}
+
+
+
+macro_rules! deref_impl {
+ ($t: ident, $ss: ident, $make_mut: expr) => {
+
+impl<T: RefArg + ?Sized> RefArg for $t<T> {
+ #[inline]
+ fn arg_type(&self) -> ArgType { (&**self).arg_type() }
+ #[inline]
+ fn signature(&self) -> Signature<'static> { (&**self).signature() }
+ #[inline]
+ fn append(&self, i: &mut IterAppend) { (&**self).append(i) }
+ #[inline]
+ fn as_any(&self) -> &any::Any where T: 'static { (&**self).as_any() }
+ #[inline]
+ fn as_any_mut<'a>(&'a mut $ss) -> &'a mut any::Any where T: 'static { $make_mut.as_any_mut() }
+ #[inline]
+ fn as_i64(&self) -> Option<i64> { (&**self).as_i64() }
+ #[inline]
+ fn as_u64(&self) -> Option<u64> { (&**self).as_u64() }
+ #[inline]
+ fn as_f64(&self) -> Option<f64> { (&**self).as_f64() }
+ #[inline]
+ fn as_str(&self) -> Option<&str> { (&**self).as_str() }
+ #[inline]
+ fn as_iter<'a>(&'a self) -> Option<Box<Iterator<Item=&'a RefArg> + 'a>> { (&**self).as_iter() }
+ #[inline]
+ fn box_clone(&self) -> Box<RefArg + 'static> { (&**self).box_clone() }
+}
+impl<T: DictKey> DictKey for $t<T> {}
+
+impl<T: Arg> Arg for $t<T> {
+ const ARG_TYPE: ArgType = T::ARG_TYPE;
+ fn signature() -> Signature<'static> { T::signature() }
+}
+impl<'a, T: Get<'a>> Get<'a> for $t<T> {
+ fn get(i: &mut Iter<'a>) -> Option<Self> { T::get(i).map(|v| $t::new(v)) }
+}
+
+ }
+}
+
+impl<T: Append> Append for Box<T> {
+ fn append(self, i: &mut IterAppend) { let q: T = *self; q.append(i) }
+}
+
+deref_impl!(Box, self, &mut **self );
+deref_impl!(Rc, self, Rc::get_mut(self).unwrap());
+deref_impl!(Arc, self, Arc::get_mut(self).unwrap());
+
+/// Internal trait to help generics. Implemented for (), (A1), (A1, A2) and so on (where A1: Arg, A2: Arg etc).
+///
+/// You would probably not use this trait directly, instead use generic functions which
+/// take ArgBuilder as an argument. It helps reading and appending multiple arguments
+/// to/from a message in one go.
+pub trait ArgBuilder: Sized {
+ /// A tuple of &static str. Used for introspection.
+ type strs;
+ /// Low-level introspection helper method.
+ fn strs_sig<F: FnMut(&'static str, Signature<'static>)>(a: Self::strs, f: F);
+ /// Low-level method to read arguments from a message.
+ fn read(msg: &Message) -> Result<Self, TypeMismatchError>;
+ /// Low-level method to append arguments to a message.
+ fn append(self, msg: &mut Message);
+}
+
+impl ArgBuilder for () {
+ type strs = ();
+ fn strs_sig<F: FnMut(&'static str, Signature<'static>)>(_: Self::strs, _: F) {}
+ fn read(_: &Message) -> Result<Self, TypeMismatchError> { Ok(()) }
+ fn append(self, _: &mut Message) {}
+}
+
+macro_rules! argbuilder_impl {
+ ($($n: ident $t: ident $s: ty,)+) => {
+
+impl<$($t: Arg + Append + for<'z> Get<'z>),*> ArgBuilder for ($($t,)*) {
+ type strs = ($(&'static $s,)*);
+ fn strs_sig<Q: FnMut(&'static str, Signature<'static>)>(z: Self::strs, mut q: Q) {
+ let ( $($n,)*) = z;
+ $( q($n, $t::signature()); )*
+ }
+
+ fn read(msg: &Message) -> Result<Self, TypeMismatchError> {
+ let mut ii = msg.iter_init();
+ $( let $n = ii.read()?; )*
+ Ok(($( $n, )* ))
+ }
+
+ fn append(self, msg: &mut Message) {
+ let ( $($n,)*) = self;
+ let mut ia = IterAppend::new(msg);
+ $( ia.append($n); )*
+ }
+}
+
+impl<$($t: Append),*> AppendAll for ($($t,)*) {
+ fn append(self, ia: &mut IterAppend) {
+ let ( $($n,)*) = self;
+ $( ia.append($n); )*
+ }
+}
+
+impl<$($t: Arg + for<'z> Get<'z>),*> ReadAll for ($($t,)*) {
+ fn read(ii: &mut Iter) -> Result<Self, TypeMismatchError> {
+ $( let $n = ii.read()?; )*
+ Ok(($( $n, )* ))
+ }
+}
+
+
+ }
+}
+
+argbuilder_impl!(a A str,);
+argbuilder_impl!(a A str, b B str,);
+argbuilder_impl!(a A str, b B str, c C str,);
+argbuilder_impl!(a A str, b B str, c C str, d D str,);
+argbuilder_impl!(a A str, b B str, c C str, d D str, e E str,);
+argbuilder_impl!(a A str, b B str, c C str, d D str, e E str, f F str,);
+argbuilder_impl!(a A str, b B str, c C str, d D str, e E str, f F str, g G str,);
+argbuilder_impl!(a A str, b B str, c C str, d D str, e E str, f F str, g G str, h H str,);
+argbuilder_impl!(a A str, b B str, c C str, d D str, e E str, f F str, g G str, h H str, i I str,);
+argbuilder_impl!(a A str, b B str, c C str, d D str, e E str, f F str, g G str, h H str, i I str, j J str,);
+
+
+
+#[cfg(test)]
+mod test {
+ extern crate tempdir;
+
+ use {Connection, ConnectionItem, Message, BusType, Path, Signature};
+ use arg::{Array, Variant, Dict, Iter, ArgType, TypeMismatchError, RefArg, cast};
+
+ use std::collections::HashMap;
+
+ #[test]
+ fn refarg() {
+ let c = Connection::get_private(BusType::Session).unwrap();
+ c.register_object_path("/mooh").unwrap();
+ let m = Message::new_method_call(&c.unique_name(), "/mooh", "com.example.hello", "Hello").unwrap();
+
+ let mut vv: Vec<Variant<Box<RefArg>>> = vec!();
+ vv.push(Variant(Box::new(5i32)));
+ vv.push(Variant(Box::new(String::from("Hello world"))));
+ let m = m.append_ref(&vv);
+
+ let (f1, f2) = (false, 7u64);
+ let mut v: Vec<&RefArg> = vec!();
+ v.push(&f1);
+ v.push(&f2);
+ let m = m.append_ref(&v);
+ let vi32 = vec![7i32, 9i32];
+ let vstr: Vec<String> = ["This", "is", "dbus", "rs"].iter().map(|&s| s.into()).collect();
+ let m = m.append_ref(&[&vi32 as &RefArg, &vstr as &RefArg]);
+ let mut map = HashMap::new();
+ map.insert(true, String::from("Yes"));
+ map.insert(false, String::from("No"));
+ let m = m.append_ref(&[&map as &RefArg, &1.5f64 as &RefArg]);
+
+ c.send(m).unwrap();
+
+ for n in c.iter(1000) {
+ if let ConnectionItem::MethodCall(m) = n {
+ let rv: Vec<Box<RefArg + 'static>> = m.iter_init().collect();
+ println!("Receiving {:?}", rv);
+ let rv0: &Variant<Box<RefArg>> = cast(&rv[0]).unwrap();
+ let rv00: &i32 = cast(&rv0.0).unwrap();
+ assert_eq!(rv00, &5i32);
+ assert_eq!(Some(&false), rv[2].as_any().downcast_ref::<bool>());
+ assert_eq!(Some(&vi32), rv[4].as_any().downcast_ref::<Vec<i32>>());
+ assert_eq!(Some(&vstr), rv[5].as_any().downcast_ref::<Vec<String>>());
+ let mut diter = rv[6].as_iter().unwrap();
+ {
+ let mut mmap: HashMap<bool, String> = HashMap::new();
+ while let Some(k) = diter.next() {
+ let x: String = diter.next().unwrap().as_str().unwrap().into();
+ mmap.insert(*cast::<bool>(&k.box_clone()).unwrap(), x);
+ }
+ assert_eq!(mmap[&true], "Yes");
+ }
+ let mut iter = rv[6].as_iter().unwrap();
+ assert!(iter.next().unwrap().as_i64().is_some());
+ assert!(iter.next().unwrap().as_str().is_some());
+ assert!(iter.next().unwrap().as_str().is_none());
+ assert!(iter.next().unwrap().as_i64().is_none());
+ assert!(iter.next().is_none());
+ assert!(rv[7].as_f64().unwrap() > 1.0);
+ assert!(rv[7].as_f64().unwrap() < 2.0);
+ break;
+ }
+ }
+ }
+
+ #[test]
+ fn message_types() {
+ let c = Connection::get_private(BusType::Session).unwrap();
+ c.register_object_path("/hello").unwrap();
+ let m = Message::new_method_call(&c.unique_name(), "/hello", "com.example.hello", "Hello").unwrap();
+ let m = m.append1(2000u16);
+ let m = m.append1(Array::new(&vec![129u8, 5, 254]));
+ let m = m.append2(Variant(&["Hello", "world"][..]), &[32768u16, 16u16, 12u16][..]);
+ let m = m.append3(-1i32, &*format!("Hello world"), -3.14f64);
+ let m = m.append1((256i16, Variant(18_446_744_073_709_551_615u64)));
+ let m = m.append2(Path::new("/a/valid/path").unwrap(), Signature::new("a{sv}").unwrap());
+ let mut z = HashMap::new();
+ z.insert(123543u32, true);
+ z.insert(0u32, false);
+ let m = m.append1(Dict::new(&z));
+ let sending = format!("{:?}", m.iter_init());
+ println!("Sending {}", sending);
+ c.send(m).unwrap();
+
+ for n in c.iter(1000) {
+ match n {
+ ConnectionItem::MethodCall(m) => {
+ use super::Arg;
+ let receiving = format!("{:?}", m.iter_init());
+ println!("Receiving {}", receiving);
+ assert_eq!(sending, receiving);
+
+ assert_eq!(2000u16, m.get1().unwrap());
+ assert_eq!(m.get2(), (Some(2000u16), Some(&[129u8, 5, 254][..])));
+ assert_eq!(m.read2::<u16, bool>().unwrap_err(),
+ TypeMismatchError { position: 1, found: ArgType::Array, expected: ArgType::Boolean });
+
+ let mut g = m.iter_init();
+ let e = g.read::<u32>().unwrap_err();
+ assert_eq!(e.pos(), 0);
+ assert_eq!(e.expected_arg_type(), ArgType::UInt32);
+ assert_eq!(e.found_arg_type(), ArgType::UInt16);
+
+ assert!(g.next() && g.next());
+ let v: Variant<Iter> = g.get().unwrap();
+ let mut viter = v.0;
+ assert_eq!(viter.arg_type(), Array::<&str,()>::ARG_TYPE);
+ let a: Array<&str, _> = viter.get().unwrap();
+ assert_eq!(a.collect::<Vec<&str>>(), vec!["Hello", "world"]);
+
+ assert!(g.next());
+ assert_eq!(g.get::<u16>(), None); // It's an array, not a single u16
+ assert!(g.next() && g.next() && g.next() && g.next());
+
+ assert_eq!(g.get(), Some((256i16, Variant(18_446_744_073_709_551_615u64))));
+ assert!(g.next());
+ assert_eq!(g.get(), Some(Path::new("/a/valid/path").unwrap()));
+ assert!(g.next());
+ assert_eq!(g.get(), Some(Signature::new("a{sv}").unwrap()));
+ assert!(g.next());
+ let d: Dict<u32, bool, _> = g.get().unwrap();
+ let z2: HashMap<_, _> = d.collect();
+ assert_eq!(z, z2);
+ break;
+ }
+ _ => println!("Got {:?}", n),
+ }
+ }
+ }
+}