use super::{ByteOrder, SigStr};
use crate::rustbus_core;
use rustbus_core::message_builder::{MarshalContext, UnmarshalContext};
use rustbus_core::wire::marshal::traits::{Marshal, SignatureBuffer};
use rustbus_core::wire::unixfd::UnixFd;
use rustbus_core::wire::unmarshal::traits::Unmarshal;
use rustbus_core::wire::validate_raw;
use std::sync::Arc;
#[derive(Debug)]
pub struct MarshalledMessageBody {
buf: Arc<Vec<u8>>,
sig: SignatureBuffer,
raw_fds: Vec<rustbus_core::wire::UnixFd>,
byteorder: ByteOrder,
}
impl Default for MarshalledMessageBody {
fn default() -> Self {
Self::new()
}
}
impl MarshalledMessageBody {
#[inline]
pub fn new() -> Self {
Self::with_byteorder(ByteOrder::LittleEndian)
}
#[inline]
pub fn with_byteorder(b: ByteOrder) -> Self {
MarshalledMessageBody {
buf: Arc::new(Vec::new()),
raw_fds: Vec::new(),
sig: SignatureBuffer::new(),
byteorder: b,
}
}
#[inline]
pub fn from_parts(
buf: Vec<u8>,
raw_fds: Vec<rustbus_core::wire::UnixFd>,
sig: SignatureBuffer,
byteorder: ByteOrder,
) -> Self {
Self {
buf: Arc::new(buf),
sig,
raw_fds,
byteorder,
}
}
#[inline]
pub fn sig(&self) -> &str {
&self.sig
}
#[inline]
pub fn buf(&self) -> &[u8] {
&self.buf
}
#[inline]
pub fn buf_arc(&self) -> Arc<Vec<u8>> {
self.buf.clone()
}
#[inline]
pub fn fds(&self) -> &[UnixFd] {
&self.raw_fds
}
#[inline]
pub fn byteorder(&self) -> ByteOrder {
self.byteorder
}
#[inline]
pub fn get_fds(&self) -> Vec<UnixFd> {
self.raw_fds.clone()
}
pub fn reset(&mut self) {
self.sig.clear();
Arc::make_mut(&mut self.buf).clear();
}
pub fn reserve(&mut self, additional: usize) {
Arc::make_mut(&mut self.buf).reserve(additional)
}
fn create_ctx(&mut self) -> MarshalContext {
MarshalContext {
buf: Arc::make_mut(&mut self.buf),
fds: &mut self.raw_fds,
byteorder: self.byteorder,
}
}
pub fn push_param<P: Marshal>(&mut self, p: P) -> Result<(), rustbus_core::Error> {
let mut ctx = self.create_ctx();
p.marshal(&mut ctx)?;
P::sig_str(&mut self.sig);
Ok(())
}
pub fn push_param2<P1: Marshal, P2: Marshal>(
&mut self,
p1: P1,
p2: P2,
) -> Result<(), rustbus_core::Error> {
let mut ctx = self.create_ctx();
let pre_len = ctx.buf.len();
let pre_fds = ctx.fds.len();
let mut marshal = || {
p1.marshal(&mut ctx)?;
p2.marshal(&mut ctx)
};
if let Err(e) = (marshal)() {
ctx.buf.truncate(pre_len);
ctx.fds.truncate(pre_fds);
return Err(e);
}
P1::sig_str(&mut self.sig);
P2::sig_str(&mut self.sig);
Ok(())
}
pub fn push_param3<P1: Marshal, P2: Marshal, P3: Marshal>(
&mut self,
p1: P1,
p2: P2,
p3: P3,
) -> Result<(), rustbus_core::Error> {
let mut ctx = self.create_ctx();
let pre_len = ctx.buf.len();
let pre_fds = ctx.fds.len();
let mut marshal = || {
p1.marshal(&mut ctx)?;
p2.marshal(&mut ctx)?;
p3.marshal(&mut ctx)
};
if let Err(e) = (marshal)() {
ctx.buf.truncate(pre_len);
ctx.fds.truncate(pre_fds);
return Err(e);
}
P1::sig_str(&mut self.sig);
P2::sig_str(&mut self.sig);
P3::sig_str(&mut self.sig);
Ok(())
}
pub fn push_param4<P1: Marshal, P2: Marshal, P3: Marshal, P4: Marshal>(
&mut self,
p1: P1,
p2: P2,
p3: P3,
p4: P4,
) -> Result<(), rustbus_core::Error> {
let mut ctx = self.create_ctx();
let pre_len = ctx.buf.len();
let pre_fds = ctx.fds.len();
let mut marshal = || {
p1.marshal(&mut ctx)?;
p2.marshal(&mut ctx)?;
p3.marshal(&mut ctx)?;
p4.marshal(&mut ctx)
};
if let Err(e) = (marshal)() {
ctx.buf.truncate(pre_len);
ctx.fds.truncate(pre_fds);
return Err(e);
}
P1::sig_str(&mut self.sig);
P2::sig_str(&mut self.sig);
P3::sig_str(&mut self.sig);
P4::sig_str(&mut self.sig);
Ok(())
}
pub fn push_param5<P1: Marshal, P2: Marshal, P3: Marshal, P4: Marshal, P5: Marshal>(
&mut self,
p1: P1,
p2: P2,
p3: P3,
p4: P4,
p5: P5,
) -> Result<(), rustbus_core::Error> {
let mut ctx = self.create_ctx();
let pre_len = ctx.buf.len();
let pre_fds = ctx.fds.len();
let mut marshal = || {
p1.marshal(&mut ctx)?;
p2.marshal(&mut ctx)?;
p3.marshal(&mut ctx)?;
p4.marshal(&mut ctx)?;
p5.marshal(&mut ctx)
};
if let Err(e) = (marshal)() {
ctx.buf.truncate(pre_len);
ctx.fds.truncate(pre_fds);
return Err(e);
}
P1::sig_str(&mut self.sig);
P2::sig_str(&mut self.sig);
P3::sig_str(&mut self.sig);
P4::sig_str(&mut self.sig);
P5::sig_str(&mut self.sig);
Ok(())
}
pub fn push_params<P: Marshal>(&mut self, params: &[P]) -> Result<(), rustbus_core::Error> {
for p in params {
self.push_param(p)?;
}
Ok(())
}
pub fn push_variant<P: Marshal>(&mut self, p: P) -> Result<(), rustbus_core::Error> {
let mut ctx = self.create_ctx();
marshal_as_variant(p, &mut ctx)
}
pub fn validate(&self) -> Result<(), rustbus_core::wire::unmarshal::Error> {
if self.sig.is_empty() && self.buf.is_empty() {
return Ok(());
}
let types = rustbus_core::signature::Type::parse_description(&self.sig)?;
let mut used = 0;
for typ in types {
used += validate_raw::validate_marshalled(self.byteorder, used, &self.buf, &typ)
.map_err(|(_, e)| e)?;
}
if used == self.buf.len() {
Ok(())
} else {
Err(rustbus_core::wire::unmarshal::Error::NotAllBytesUsed)
}
}
#[inline]
pub fn parser(&self) -> MessageBodyParser {
MessageBodyParser::new(&self)
}
}
#[derive(Debug)]
pub struct MessageBodyParser<'body> {
buf_idx: usize,
sig_idx: usize,
sigs: Vec<rustbus_core::signature::Type>,
body: &'body MarshalledMessageBody,
}
impl<'ret, 'fds, 'body: 'ret + 'fds> MessageBodyParser<'body> {
pub fn new(body: &'body MarshalledMessageBody) -> Self {
let sigs = match rustbus_core::signature::Type::parse_description(&body.sig) {
Ok(sigs) => sigs,
Err(e) => match e {
rustbus_core::signature::Error::EmptySignature => Vec::new(),
_ => panic!("MarshalledMessageBody has bad signature: {:?}", e),
},
};
Self {
buf_idx: 0,
sig_idx: 0,
sigs,
body,
}
}
pub fn get_next_sig(&self) -> Option<&rustbus_core::signature::Type> {
self.sigs.get(self.sig_idx)
}
pub fn get_left_sigs(&self) -> Option<&[rustbus_core::signature::Type]> {
self.sigs.get(self.sig_idx..)
}
pub fn get<T: Unmarshal<'body, 'fds>>(
&mut self,
) -> Result<T, rustbus_core::wire::unmarshal::Error> {
if self.sig_idx >= self.sigs.len() {
return Err(rustbus_core::wire::unmarshal::Error::EndOfMessage);
}
if self.sigs[self.sig_idx] != T::signature() {
return Err(rustbus_core::wire::unmarshal::Error::WrongSignature);
}
let mut ctx = UnmarshalContext {
byteorder: self.body.byteorder,
buf: &self.body.buf,
offset: self.buf_idx,
fds: &self.body.raw_fds,
};
match T::unmarshal(&mut ctx) {
Ok((bytes, res)) => {
self.buf_idx += bytes;
self.sig_idx += 1;
Ok(res)
}
Err(e) => Err(e),
}
}
fn get_mult_helper<T, F>(
&mut self,
count: usize,
get_calls: F,
) -> Result<T, rustbus_core::wire::unmarshal::Error>
where
F: FnOnce(&mut Self) -> Result<T, rustbus_core::wire::unmarshal::Error>,
{
if self.sig_idx + count > self.sigs.len() {
return Err(rustbus_core::wire::unmarshal::Error::EndOfMessage);
}
let start_sig_idx = self.sig_idx;
let start_buf_idx = self.buf_idx;
match get_calls(self) {
Ok(ret) => Ok(ret),
Err(err) => {
self.sig_idx = start_sig_idx;
self.buf_idx = start_buf_idx;
Err(err)
}
}
}
pub fn get2<T1, T2>(&mut self) -> Result<(T1, T2), rustbus_core::wire::unmarshal::Error>
where
T1: Unmarshal<'body, 'fds>,
T2: Unmarshal<'body, 'fds>,
{
let get_calls = |parser: &mut Self| {
let ret1 = parser.get()?;
let ret2 = parser.get()?;
Ok((ret1, ret2))
};
self.get_mult_helper(2, get_calls)
}
pub fn get3<T1, T2, T3>(&mut self) -> Result<(T1, T2, T3), rustbus_core::wire::unmarshal::Error>
where
T1: Unmarshal<'body, 'fds>,
T2: Unmarshal<'body, 'fds>,
T3: Unmarshal<'body, 'fds>,
{
let get_calls = |parser: &mut Self| {
let ret1 = parser.get()?;
let ret2 = parser.get()?;
let ret3 = parser.get()?;
Ok((ret1, ret2, ret3))
};
self.get_mult_helper(3, get_calls)
}
pub fn get4<T1, T2, T3, T4>(
&mut self,
) -> Result<(T1, T2, T3, T4), rustbus_core::wire::unmarshal::Error>
where
T1: Unmarshal<'body, 'fds>,
T2: Unmarshal<'body, 'fds>,
T3: Unmarshal<'body, 'fds>,
T4: Unmarshal<'body, 'fds>,
{
let get_calls = |parser: &mut Self| {
let ret1 = parser.get()?;
let ret2 = parser.get()?;
let ret3 = parser.get()?;
let ret4 = parser.get()?;
Ok((ret1, ret2, ret3, ret4))
};
self.get_mult_helper(4, get_calls)
}
pub fn get5<T1, T2, T3, T4, T5>(
&mut self,
) -> Result<(T1, T2, T3, T4, T5), rustbus_core::wire::unmarshal::Error>
where
T1: Unmarshal<'body, 'fds>,
T2: Unmarshal<'body, 'fds>,
T3: Unmarshal<'body, 'fds>,
T4: Unmarshal<'body, 'fds>,
T5: Unmarshal<'body, 'fds>,
{
let get_calls = |parser: &mut Self| {
let ret1 = parser.get()?;
let ret2 = parser.get()?;
let ret3 = parser.get()?;
let ret4 = parser.get()?;
let ret5 = parser.get()?;
Ok((ret1, ret2, ret3, ret4, ret5))
};
self.get_mult_helper(5, get_calls)
}
}
fn marshal_as_variant<T: Marshal>(
val: T,
ctx: &mut MarshalContext,
) -> Result<(), rustbus_core::Error> {
let sig = T::signature();
let mut s_str = String::with_capacity(255);
sig.to_str(&mut s_str);
debug_assert!(SigStr::new(&s_str).is_ok());
let ss = unsafe { SigStr::new_no_val(&s_str) };
ss.marshal(ctx).unwrap();
val.marshal(ctx)
}
#[test]
fn test_marshal_trait() {
let mut body = MarshalledMessageBody::new();
let bytes: &[&[_]] = &[&[4u64]];
body.push_param(bytes).unwrap();
assert_eq!(
[12, 0, 0, 0, 8, 0, 0, 0, 4, 0, 0, 0, 0, 0, 0, 0],
&body.buf[..]
);
assert_eq!(body.sig.as_str(), "aat");
let mut body = MarshalledMessageBody::new();
let mut map = std::collections::HashMap::new();
map.insert("a", 4u32);
body.push_param(&map).unwrap();
assert_eq!(
[12, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, b'a', 0, 0, 0, 4, 0, 0, 0,],
&body.buf[..]
);
assert_eq!(body.sig.as_str(), "a{su}");
let mut body = MarshalledMessageBody::new();
body.push_param((11u64, "str", true)).unwrap();
assert_eq!(body.sig.as_str(), "(tsb)");
assert_eq!(
[11, 0, 0, 0, 0, 0, 0, 0, 3, 0, 0, 0, b's', b't', b'r', 0, 1, 0, 0, 0,],
&body.buf[..]
);
struct MyStruct {
x: u64,
y: String,
}
use rustbus_core::wire::marshal::traits::Signature;
use rustbus_core::wire::marshal::MarshalContext;
impl Signature for &MyStruct {
fn signature() -> rustbus_core::signature::Type {
rustbus_core::signature::Type::Container(rustbus_core::signature::Container::Struct(
rustbus_core::signature::StructTypes::new(vec![
u64::signature(),
String::signature(),
])
.unwrap(),
))
}
fn alignment() -> usize {
8
}
}
impl Marshal for &MyStruct {
fn marshal(&self, ctx: &mut MarshalContext) -> Result<(), rustbus_core::Error> {
ctx.align_to(8);
self.x.marshal(ctx)?;
self.y.marshal(ctx)?;
Ok(())
}
}
let mut body = MarshalledMessageBody::new();
body.push_param(&MyStruct {
x: 100,
y: "A".to_owned(),
})
.unwrap();
assert_eq!(body.sig.as_str(), "(ts)");
assert_eq!(
[100, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, b'A', 0,],
&body.buf[..]
);
let mut body = MarshalledMessageBody::new();
let emptymap: std::collections::HashMap<&str, u32> = std::collections::HashMap::new();
let mut map = std::collections::HashMap::new();
let mut map2 = std::collections::HashMap::new();
map.insert("a", 4u32);
map2.insert("a", &map);
body.push_param(&map2).unwrap();
body.push_param(&emptymap).unwrap();
assert_eq!(body.sig.as_str(), "a{sa{su}}a{su}");
assert_eq!(
[
28, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, b'a', 0, 0, 0, 12, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0,
0, b'a', 0, 0, 0, 4, 0, 0, 0, 0, 0, 0, 0
],
&body.buf[..]
);
let mut body_iter = MessageBodyParser::new(&body);
type WrongNestedDict =
std::collections::HashMap<String, std::collections::HashMap<String, u64>>;
assert_eq!(
body_iter.get::<WrongNestedDict>().err().unwrap(),
rustbus_core::wire::unmarshal::Error::WrongSignature
);
type WrongStruct = (u64, i32, String);
assert_eq!(
body_iter.get::<WrongStruct>().err().unwrap(),
rustbus_core::wire::unmarshal::Error::WrongSignature
);
type NestedDict = std::collections::HashMap<String, std::collections::HashMap<String, u32>>;
let newmap2: NestedDict = body_iter.get().unwrap();
assert_eq!(newmap2.len(), 1);
assert_eq!(newmap2.get("a").unwrap().len(), 1);
assert_eq!(*newmap2.get("a").unwrap().get("a").unwrap(), 4);
assert_eq!(
body_iter.get::<WrongNestedDict>().err().unwrap(),
rustbus_core::wire::unmarshal::Error::WrongSignature
);
assert_eq!(
body_iter.get::<WrongStruct>().err().unwrap(),
rustbus_core::wire::unmarshal::Error::WrongSignature
);
let newemptymap: std::collections::HashMap<&str, u32> = body_iter.get().unwrap();
assert_eq!(newemptymap.len(), 0);
let mut body_iter = body.parser();
assert_eq!(
body_iter.get2::<NestedDict, u16>().unwrap_err(),
rustbus_core::wire::unmarshal::Error::WrongSignature
);
assert_eq!(
body_iter
.get3::<NestedDict, std::collections::HashMap<&str, u32>, u32>()
.unwrap_err(),
rustbus_core::wire::unmarshal::Error::EndOfMessage
);
let (newmap2, newemptymap): (NestedDict, std::collections::HashMap<&str, u32>) =
body_iter.get2().unwrap();
assert_eq!(newmap2.len(), 1);
assert_eq!(newmap2.get("a").unwrap().len(), 1);
assert_eq!(*newmap2.get("a").unwrap().get("a").unwrap(), 4);
assert_eq!(newemptymap.len(), 0);
assert_eq!(
body_iter.get::<u16>().unwrap_err(),
rustbus_core::wire::unmarshal::Error::EndOfMessage
);
let mut body_iter = body.parser();
let newmap2: NestedDict = body_iter.get().unwrap();
assert_eq!(newmap2.len(), 1);
assert_eq!(newmap2.get("a").unwrap().len(), 1);
assert_eq!(*newmap2.get("a").unwrap().get("a").unwrap(), 4);
}