mirror of
https://github.com/cloud-hypervisor/cloud-hypervisor.git
synced 2025-01-09 14:25:21 +00:00
a8f063db7c
Refactor the serial buffer handling in order to write the serial buffer's output to a PTY connected after the serial device stops being written to by the guest. This change moves the serial buffer initialization inside the serial manager. That is done to allow the serial buffer to be made aware of the PTY and epoll fds needed in order to modify the EpollDispatch::File trigger. These are then used by the serial buffer to trigger an epoll event when the PTY fd is writable and the buffer has content in it. They are also used to remove the trigger when the buffer is emptied in order to avoid unnecessary wake-ups. Signed-off-by: William Douglas <william.douglas@intel.com>
176 lines
5.3 KiB
Rust
176 lines
5.3 KiB
Rust
// Copyright © 2021 Intel Corporation
|
|
//
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
//
|
|
|
|
use crate::serial_manager::EpollDispatch;
|
|
|
|
use std::io::Write;
|
|
use std::os::unix::io::RawFd;
|
|
|
|
// Circular buffer implementation for serial output.
|
|
// Read from head; push to tail
|
|
pub(crate) struct SerialBuffer {
|
|
buffer: Vec<u8>,
|
|
head: usize,
|
|
tail: usize,
|
|
out: Box<dyn Write + Send>,
|
|
buffering: bool,
|
|
out_fd: Option<RawFd>,
|
|
epoll_fd: Option<RawFd>,
|
|
}
|
|
|
|
const MAX_BUFFER_SIZE: usize = 16 << 10;
|
|
|
|
impl SerialBuffer {
|
|
pub(crate) fn new(out: Box<dyn Write + Send>) -> Self {
|
|
Self {
|
|
buffer: vec![],
|
|
head: 0,
|
|
tail: 0,
|
|
out,
|
|
buffering: false,
|
|
out_fd: None,
|
|
epoll_fd: None,
|
|
}
|
|
}
|
|
|
|
pub(crate) fn add_out_fd(&mut self, out_fd: RawFd) {
|
|
self.out_fd = Some(out_fd);
|
|
}
|
|
|
|
pub(crate) fn add_epoll_fd(&mut self, epoll_fd: RawFd) {
|
|
self.epoll_fd = Some(epoll_fd);
|
|
}
|
|
|
|
pub fn flush_buffer(&mut self) -> Result<(), std::io::Error> {
|
|
if self.tail <= self.head {
|
|
// The buffer to be written is in two parts
|
|
let buf = &self.buffer[self.head..];
|
|
match self.out.write(buf) {
|
|
Ok(bytes_written) => {
|
|
if bytes_written == buf.len() {
|
|
self.head = 0;
|
|
// Can now proceed to write the other part of the buffer
|
|
} else {
|
|
self.head += bytes_written;
|
|
self.out.flush()?;
|
|
return Ok(());
|
|
}
|
|
}
|
|
Err(e) => {
|
|
if !matches!(e.kind(), std::io::ErrorKind::WouldBlock) {
|
|
return Err(e);
|
|
}
|
|
self.add_out_poll()?;
|
|
return Ok(());
|
|
}
|
|
}
|
|
}
|
|
|
|
let buf = &self.buffer[self.head..self.tail];
|
|
match self.out.write(buf) {
|
|
Ok(bytes_written) => {
|
|
if bytes_written == buf.len() {
|
|
self.buffer.clear();
|
|
self.buffer.shrink_to_fit();
|
|
self.head = 0;
|
|
self.tail = 0;
|
|
self.remove_out_poll()?;
|
|
} else {
|
|
self.head += bytes_written;
|
|
}
|
|
self.out.flush()?;
|
|
}
|
|
Err(e) => {
|
|
if !matches!(e.kind(), std::io::ErrorKind::WouldBlock) {
|
|
return Err(e);
|
|
}
|
|
self.add_out_poll()?;
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn add_out_poll(&mut self) -> Result<(), std::io::Error> {
|
|
if self.out_fd.is_some() && self.epoll_fd.is_some() && !self.buffering {
|
|
self.buffering = true;
|
|
let out_fd = self.out_fd.as_ref().unwrap();
|
|
let epoll_fd = self.epoll_fd.as_ref().unwrap();
|
|
epoll::ctl(
|
|
*epoll_fd,
|
|
epoll::ControlOptions::EPOLL_CTL_MOD,
|
|
*out_fd,
|
|
epoll::Event::new(
|
|
epoll::Events::EPOLLIN | epoll::Events::EPOLLOUT,
|
|
EpollDispatch::File as u64,
|
|
),
|
|
)?;
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
fn remove_out_poll(&mut self) -> Result<(), std::io::Error> {
|
|
if self.out_fd.is_some() && self.epoll_fd.is_some() && self.buffering {
|
|
self.buffering = false;
|
|
let out_fd = self.out_fd.as_ref().unwrap();
|
|
let epoll_fd = self.epoll_fd.as_ref().unwrap();
|
|
epoll::ctl(
|
|
*epoll_fd,
|
|
epoll::ControlOptions::EPOLL_CTL_MOD,
|
|
*out_fd,
|
|
epoll::Event::new(epoll::Events::EPOLLIN, EpollDispatch::File as u64),
|
|
)?;
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
impl Write for SerialBuffer {
|
|
fn write(&mut self, buf: &[u8]) -> Result<usize, std::io::Error> {
|
|
// The serial output only writes one byte at a time
|
|
for v in buf {
|
|
if self.buffer.is_empty() {
|
|
// This case exists to avoid allocating the buffer if it's not needed
|
|
if let Err(e) = self.out.write(&[*v]) {
|
|
if !matches!(e.kind(), std::io::ErrorKind::WouldBlock) {
|
|
return Err(e);
|
|
}
|
|
self.add_out_poll()?;
|
|
self.buffer.push(*v);
|
|
self.tail += 1;
|
|
} else {
|
|
self.out.flush()?;
|
|
}
|
|
} else {
|
|
// Buffer is completely full, lose the oldest byte by moving head forward
|
|
if self.head == self.tail {
|
|
self.head = self.tail + 1;
|
|
if self.head == MAX_BUFFER_SIZE {
|
|
self.head = 0;
|
|
}
|
|
}
|
|
|
|
if self.buffer.len() < MAX_BUFFER_SIZE {
|
|
self.buffer.push(*v);
|
|
} else {
|
|
self.buffer[self.tail] = *v;
|
|
}
|
|
|
|
self.tail += 1;
|
|
if self.tail == MAX_BUFFER_SIZE {
|
|
self.tail = 0;
|
|
}
|
|
|
|
self.flush_buffer()?;
|
|
}
|
|
}
|
|
|
|
Ok(buf.len())
|
|
}
|
|
|
|
fn flush(&mut self) -> Result<(), std::io::Error> {
|
|
self.flush_buffer()
|
|
}
|
|
}
|