use crate::error::{Error, Result}; use crate::sys::{XsdMessageHeader, XSD_ERROR}; use std::ffi::CString; use std::fs::{self, metadata, File}; use std::io::{Read, Write}; use std::mem::size_of; use std::os::unix::fs::FileTypeExt; use std::os::unix::net::UnixStream; const XEN_BUS_PATHS: &[&str] = &["/dev/xen/xenbus", "/var/run/xenstored/socket"]; fn find_bus_path() -> Option { for path in XEN_BUS_PATHS { match metadata(path) { Ok(_) => return Some(String::from(*path)), Err(_) => continue, } } None } trait XsdTransport { fn xsd_write_all(&mut self, buf: &[u8]) -> Result<()>; fn xsd_read_exact(&mut self, buf: &mut [u8]) -> Result<()>; } impl XsdTransport for UnixStream { fn xsd_write_all(&mut self, buf: &[u8]) -> Result<()> { Ok(self.write_all(buf)?) } fn xsd_read_exact(&mut self, buf: &mut [u8]) -> Result<()> { Ok(self.read_exact(buf)?) } } pub struct XsdFileTransport { handle: File, } impl XsdFileTransport { pub fn new(path: &str) -> Result { let handle = File::options().read(true).write(true).open(path)?; Ok(XsdFileTransport { handle }) } } impl XsdTransport for XsdFileTransport { fn xsd_read_exact(&mut self, buf: &mut [u8]) -> Result<()> { Ok(self.handle.read_exact(buf)?) } fn xsd_write_all(&mut self, buf: &[u8]) -> Result<()> { self.handle.write_all(buf)?; self.handle.flush()?; Ok(()) } } pub struct XsdSocket { handle: Box, } #[derive(Debug)] pub struct XsdResponse { pub header: XsdMessageHeader, pub payload: Vec, } impl XsdResponse { pub fn parse_string(&self) -> Result { Ok(CString::from_vec_with_nul(self.payload.clone())?.into_string()?) } pub fn parse_string_vec(&self) -> Result> { let mut strings: Vec = Vec::new(); let mut buffer: Vec = Vec::new(); for b in &self.payload { if *b == 0 { let string = String::from_utf8(buffer.clone())?; strings.push(string); buffer.clear(); continue; } buffer.push(*b); } Ok(strings) } pub fn parse_bool(&self) -> Result { Ok(true) } } impl XsdSocket { pub fn dial() -> Result { let path = match find_bus_path() { Some(path) => path, None => return Err(Error::BusNotFound), }; let metadata = fs::metadata(&path)?; let file_type = metadata.file_type(); if file_type.is_socket() { let stream = UnixStream::connect(&path)?; return Ok(XsdSocket { handle: Box::new(stream), }); } let transport = XsdFileTransport::new(&path)?; Ok(XsdSocket { handle: Box::new(transport), }) } pub fn send(&mut self, tx: u32, typ: u32, buf: &[u8]) -> Result { let header = XsdMessageHeader { typ, req: 0, tx, len: buf.len() as u32, }; let header_bytes = bytemuck::bytes_of(&header); let mut composed: Vec = Vec::new(); composed.extend_from_slice(header_bytes); composed.extend_from_slice(buf); self.handle.xsd_write_all(&composed)?; let mut result_buf = vec![0u8; size_of::()]; self.handle.xsd_read_exact(result_buf.as_mut_slice())?; let result_header = bytemuck::from_bytes::(&result_buf); let mut payload = vec![0u8; result_header.len as usize]; self.handle.xsd_read_exact(payload.as_mut_slice())?; if result_header.typ == XSD_ERROR { let error = CString::from_vec_with_nul(payload)?; return Err(Error::ResponseError(error.into_string()?)); } let response = XsdResponse { header, payload }; Ok(response) } pub fn send_single(&mut self, tx: u32, typ: u32, string: &str) -> Result { let text = CString::new(string)?; let buf = text.as_bytes_with_nul(); self.send(tx, typ, buf) } pub fn send_multiple(&mut self, tx: u32, typ: u32, array: &[&str]) -> Result { let mut buf: Vec = Vec::new(); for item in array { buf.extend_from_slice(item.as_bytes()); buf.push(0); } self.send(tx, typ, buf.as_slice()) } }