opentitanlib/transport/proxy/
mod.rsuse anyhow::{bail, Context, Result};
use serde::{Deserialize, Serialize};
use std::cell::{Cell, RefCell};
use std::collections::HashMap;
use std::io;
use std::io::{BufWriter, ErrorKind, Read, Write};
use std::net::{TcpStream, ToSocketAddrs};
use std::os::unix::io::AsRawFd;
use std::rc::Rc;
use std::time::Duration;
use thiserror::Error;
use crate::bootstrap::BootstrapOptions;
use crate::impl_serializable_error;
use crate::io::emu::Emulator;
use crate::io::gpio::{GpioBitbanging, GpioMonitoring, GpioPin};
use crate::io::i2c::Bus;
use crate::io::nonblocking_help::NonblockingHelp;
use crate::io::spi::Target;
use crate::io::uart::Uart;
use crate::proxy::protocol::{
AsyncMessage, Message, ProxyRequest, ProxyResponse, Request, Response,
};
use crate::transport::{Capabilities, Capability, ProxyOps, Transport, TransportError};
mod emu;
mod gpio;
mod i2c;
mod spi;
mod uart;
#[derive(Debug, Error, Serialize, Deserialize)]
pub enum ProxyError {
#[error("Unexpected reply")]
UnexpectedReply(),
#[error("JSON encoding: {0}")]
JsonEncoding(String),
#[error("JSON decoding: {0}")]
JsonDecoding(String),
}
impl_serializable_error!(ProxyError);
pub struct Proxy {
inner: Rc<Inner>,
}
impl Proxy {
pub fn open(host: Option<&str>, port: u16) -> Result<Self> {
let host = host.unwrap_or("localhost");
let addr = ToSocketAddrs::to_socket_addrs(&(host, port))
.map_err(|e| TransportError::ProxyLookupError(host.to_string(), e.to_string()))?
.next()
.unwrap();
let conn = TcpStream::connect(addr)
.map_err(|e| TransportError::ProxyConnectError(addr.to_string(), e.to_string()))?;
Ok(Self {
inner: Rc::new(Inner {
conn: RefCell::new(conn),
uarts: RefCell::new(HashMap::new()),
uart_channel_map: RefCell::new(HashMap::new()),
recv_buf: RefCell::new(Vec::new()),
nonblocking_help_enabled: Cell::new(false),
}),
})
}
}
struct UartRecord {
pub uart: Rc<dyn Uart>,
pub pipe_sender: mio::unix::pipe::Sender,
pub pipe_receiver: mio::unix::pipe::Receiver,
}
struct Inner {
conn: RefCell<TcpStream>,
pub uarts: RefCell<HashMap<String, UartRecord>>,
uart_channel_map: RefCell<HashMap<u32, String>>,
recv_buf: RefCell<Vec<u8>>,
nonblocking_help_enabled: Cell<bool>,
}
impl Inner {
fn execute_command(&self, req: Request) -> Result<Response> {
self.send_json_request(req).context("json encoding")?;
loop {
match self.recv_json_response().context("json decoding")? {
Message::Res(res) => match res {
Ok(value) => return Ok(value),
Err(e) => return Err(anyhow::Error::from(e)),
},
Message::Async { channel, msg } => self.process_async_data(channel, msg)?,
_ => bail!(ProxyError::UnexpectedReply()),
}
}
}
fn poll_for_async_data(&self, timeout: Option<Duration>) -> Result<()> {
if timeout == Some(Duration::from_millis(0)) {
self.recv_nonblocking()?;
} else {
self.recv_with_timeout(timeout)?;
}
while let Some(msg) = self.dequeue_json_response()? {
match msg {
Message::Async { channel, msg } => self.process_async_data(channel, msg)?,
_ => bail!(ProxyError::UnexpectedReply()),
}
}
Ok(())
}
fn process_async_data(&self, channel: u32, msg: AsyncMessage) -> Result<()> {
match msg {
AsyncMessage::UartData { data } => {
if let Some(uart_instance) = self.uart_channel_map.borrow().get(&channel) {
if let Some(uart_record) = self.uarts.borrow_mut().get_mut(uart_instance) {
uart_record.pipe_sender.write_all(&data)?;
}
}
}
}
Ok(())
}
fn send_json_request(&self, req: Request) -> Result<()> {
let conn: &mut std::net::TcpStream = &mut self.conn.borrow_mut();
let mut writer = BufWriter::new(conn);
serde_json::to_writer(&mut writer, &Message::Req(req))?;
writer.write_all(b"\n")?;
writer.flush()?;
Ok(())
}
fn recv_json_response(&self) -> Result<Message> {
if let Some(msg) = self.dequeue_json_response()? {
return Ok(msg);
}
let mut conn = self.conn.borrow_mut();
let mut buf = self.recv_buf.borrow_mut();
let mut idx: usize = buf.len();
loop {
buf.resize(idx + 2048, 0);
let rc = conn.read(&mut buf[idx..])?;
if rc == 0 {
anyhow::bail!(io::Error::new(
ErrorKind::UnexpectedEof,
"Server unexpectedly closed connection"
))
}
idx += rc;
let Some(newline_pos) = buf[idx - rc..idx].iter().position(|b| *b == b'\n') else {
continue;
};
let result = serde_json::from_slice::<Message>(&buf[..idx - rc + newline_pos])?;
buf.resize(idx, 0u8);
buf.drain(..idx - rc + newline_pos + 1);
return Ok(result);
}
}
fn recv_with_timeout(&self, timeout: Option<Duration>) -> Result<()> {
let mut conn = self.conn.borrow_mut();
conn.set_read_timeout(timeout)?;
let mut buf = self.recv_buf.borrow_mut();
let mut idx: usize = buf.len();
buf.resize(idx + 2048, 0);
match conn.read(&mut buf[idx..]) {
Ok(0) => {
anyhow::bail!(io::Error::new(
ErrorKind::UnexpectedEof,
"Server unexpectedly closed connection"
))
}
Ok(rc) => idx += rc,
Err(ref e) if e.kind() == ErrorKind::WouldBlock => (),
Err(e) => anyhow::bail!(e),
}
buf.resize(idx, 0);
conn.set_read_timeout(None)?;
Ok(())
}
fn recv_nonblocking(&self) -> Result<()> {
let mut conn = self.conn.borrow_mut();
conn.set_nonblocking(true)?;
let mut buf = self.recv_buf.borrow_mut();
let mut idx: usize = buf.len();
loop {
buf.resize(idx + 2048, 0);
match conn.read(&mut buf[idx..]) {
Ok(0) => {
anyhow::bail!(io::Error::new(
ErrorKind::UnexpectedEof,
"Server unexpectedly closed connection"
))
}
Ok(rc) => idx += rc,
Err(ref e) if e.kind() == ErrorKind::WouldBlock => break,
Err(e) => anyhow::bail!(e),
}
}
buf.resize(idx, 0);
conn.set_nonblocking(false)?;
Ok(())
}
fn dequeue_json_response(&self) -> Result<Option<Message>> {
let mut buf = self.recv_buf.borrow_mut();
let Some(newline_pos) = buf.iter().position(|b| *b == b'\n') else {
return Ok(None);
};
let result = serde_json::from_slice::<Message>(&buf[..newline_pos])?;
buf.drain(..newline_pos + 1);
Ok(Some(result))
}
}
pub struct ProxyOpsImpl {
inner: Rc<Inner>,
}
impl ProxyOpsImpl {
pub fn new(proxy: &Proxy) -> Result<Self> {
Ok(Self {
inner: Rc::clone(&proxy.inner),
})
}
fn execute_command(&self, command: ProxyRequest) -> Result<ProxyResponse> {
match self.inner.execute_command(Request::Proxy(command))? {
Response::Proxy(resp) => Ok(resp),
_ => bail!(ProxyError::UnexpectedReply()),
}
}
}
impl ProxyOps for ProxyOpsImpl {
fn provides_map(&self) -> Result<HashMap<String, String>> {
match self.execute_command(ProxyRequest::Provides {})? {
ProxyResponse::Provides { provides_map } => Ok(provides_map),
_ => bail!(ProxyError::UnexpectedReply()),
}
}
fn bootstrap(&self, options: &BootstrapOptions, payload: &[u8]) -> Result<()> {
match self.execute_command(ProxyRequest::Bootstrap {
options: options.clone(),
payload: payload.to_vec(),
})? {
ProxyResponse::Bootstrap => Ok(()),
_ => bail!(ProxyError::UnexpectedReply()),
}
}
fn apply_pin_strapping(&self, strapping_name: &str) -> Result<()> {
match self.execute_command(ProxyRequest::ApplyPinStrapping {
strapping_name: strapping_name.to_string(),
})? {
ProxyResponse::ApplyPinStrapping => Ok(()),
_ => bail!(ProxyError::UnexpectedReply()),
}
}
fn remove_pin_strapping(&self, strapping_name: &str) -> Result<()> {
match self.execute_command(ProxyRequest::RemovePinStrapping {
strapping_name: strapping_name.to_string(),
})? {
ProxyResponse::RemovePinStrapping => Ok(()),
_ => bail!(ProxyError::UnexpectedReply()),
}
}
fn apply_default_configuration_with_strap(&self, strapping_name: &str) -> Result<()> {
match self.execute_command(ProxyRequest::ApplyDefaultConfigurationWithStrapping {
strapping_name: strapping_name.to_string(),
})? {
ProxyResponse::ApplyDefaultConfigurationWithStrapping => Ok(()),
_ => bail!(ProxyError::UnexpectedReply()),
}
}
}
impl Transport for Proxy {
fn capabilities(&self) -> Result<Capabilities> {
match self.inner.execute_command(Request::GetCapabilities)? {
Response::GetCapabilities(capabilities) => Ok(capabilities.add(Capability::PROXY)),
_ => bail!(ProxyError::UnexpectedReply()),
}
}
fn apply_default_configuration(&self) -> Result<()> {
match self
.inner
.execute_command(Request::ApplyDefaultConfiguration)?
{
Response::ApplyDefaultConfiguration => Ok(()),
_ => bail!(ProxyError::UnexpectedReply()),
}
}
fn spi(&self, instance: &str) -> Result<Rc<dyn Target>> {
Ok(Rc::new(spi::ProxySpi::open(self, instance)?))
}
fn i2c(&self, instance: &str) -> Result<Rc<dyn Bus>> {
Ok(Rc::new(i2c::ProxyI2c::open(self, instance)?))
}
fn uart(&self, instance_name: &str) -> Result<Rc<dyn Uart>> {
if let Some(instance) = self.inner.uarts.borrow().get(instance_name) {
return Ok(Rc::clone(&instance.uart));
}
let instance: Rc<dyn Uart> = Rc::new(uart::ProxyUart::open(self, instance_name)?);
let (pipe_sender, pipe_receiver) = mio::unix::pipe::new()?;
self.inner.uarts.borrow_mut().insert(
instance_name.to_string(),
UartRecord {
uart: Rc::clone(&instance),
pipe_sender,
pipe_receiver,
},
);
Ok(instance)
}
fn gpio_pin(&self, pinname: &str) -> Result<Rc<dyn GpioPin>> {
Ok(Rc::new(gpio::ProxyGpioPin::open(self, pinname)?))
}
fn gpio_monitoring(&self) -> Result<Rc<dyn GpioMonitoring>> {
Ok(Rc::new(gpio::GpioMonitoringImpl::new(self)?))
}
fn gpio_bitbanging(&self) -> Result<Rc<dyn GpioBitbanging>> {
Ok(Rc::new(gpio::GpioBitbangingImpl::new(self)?))
}
fn emulator(&self) -> Result<Rc<dyn Emulator>> {
Ok(Rc::new(emu::ProxyEmu::open(self)?))
}
fn proxy_ops(&self) -> Result<Rc<dyn ProxyOps>> {
Ok(Rc::new(ProxyOpsImpl::new(self)?))
}
fn nonblocking_help(&self) -> Result<Rc<dyn NonblockingHelp>> {
Ok(Rc::new(ProxyNonblockingHelp {
inner: self.inner.clone(),
}))
}
}
pub struct ProxyNonblockingHelp {
inner: Rc<Inner>,
}
impl NonblockingHelp for ProxyNonblockingHelp {
fn register_nonblocking_help(&self, registry: &mio::Registry, token: mio::Token) -> Result<()> {
let conn: &mut std::net::TcpStream = &mut self.inner.conn.borrow_mut();
registry.register(
&mut mio::unix::SourceFd(&conn.as_raw_fd()),
token,
mio::Interest::READABLE,
)?;
self.inner.nonblocking_help_enabled.set(true);
Ok(())
}
fn nonblocking_help(&self) -> Result<()> {
self.inner
.poll_for_async_data(Some(Duration::from_millis(0)))
}
}