Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Make vsock buffer sizes configurable #139

Merged
merged 3 commits into from
Jun 6, 2024
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
48 changes: 37 additions & 11 deletions src/device/socket/connectionmanager.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
use super::{
protocol::VsockAddr, vsock::ConnectionInfo, DisconnectReason, SocketError, VirtIOSocket,
VsockEvent, VsockEventType,
VsockEvent, VsockEventType, DEFAULT_RX_BUFFER_SIZE,
};
use crate::{transport::Transport, Hal, Result};
use alloc::{boxed::Box, vec::Vec};
Expand All @@ -10,12 +10,15 @@ use core::hint::spin_loop;
use log::debug;
use zerocopy::FromZeroes;

const PER_CONNECTION_BUFFER_CAPACITY: usize = 1024;
const DEFAULT_PER_CONNECTION_BUFFER_CAPACITY: usize = 1024;

/// A higher level interface for VirtIO socket (vsock) devices.
///
/// This keeps track of multiple vsock connections.
///
/// `RX_BUFFER_SIZE` is the size in bytes of each buffer used in the RX virtqueue. This must be
/// bigger than `size_of::<VirtioVsockHdr>()`.
///
/// # Example
///
/// ```
Expand All @@ -40,8 +43,13 @@ const PER_CONNECTION_BUFFER_CAPACITY: usize = 1024;
/// # Ok(())
/// # }
/// ```
pub struct VsockConnectionManager<H: Hal, T: Transport> {
driver: VirtIOSocket<H, T>,
pub struct VsockConnectionManager<
H: Hal,
T: Transport,
const RX_BUFFER_SIZE: usize = DEFAULT_RX_BUFFER_SIZE,
> {
driver: VirtIOSocket<H, T, RX_BUFFER_SIZE>,
per_connection_buffer_capacity: usize,
connections: Vec<Connection>,
listening_ports: Vec<u32>,
}
Expand All @@ -56,24 +64,36 @@ struct Connection {
}

impl Connection {
fn new(peer: VsockAddr, local_port: u32) -> Self {
fn new(peer: VsockAddr, local_port: u32, buffer_capacity: usize) -> Self {
let mut info = ConnectionInfo::new(peer, local_port);
info.buf_alloc = PER_CONNECTION_BUFFER_CAPACITY.try_into().unwrap();
info.buf_alloc = buffer_capacity.try_into().unwrap();
Self {
info,
buffer: RingBuffer::new(PER_CONNECTION_BUFFER_CAPACITY),
buffer: RingBuffer::new(buffer_capacity),
peer_requested_shutdown: false,
}
}
}

impl<H: Hal, T: Transport> VsockConnectionManager<H, T> {
impl<H: Hal, T: Transport, const RX_BUFFER_SIZE: usize>
VsockConnectionManager<H, T, RX_BUFFER_SIZE>
{
/// Construct a new connection manager wrapping the given low-level VirtIO socket driver.
pub fn new(driver: VirtIOSocket<H, T>) -> Self {
pub fn new(driver: VirtIOSocket<H, T, RX_BUFFER_SIZE>) -> Self {
Self::new_with_capacity(driver, DEFAULT_PER_CONNECTION_BUFFER_CAPACITY)
}

/// Construct a new connection manager wrapping the given low-level VirtIO socket driver, with
/// the given per-connection buffer capacity.
pub fn new_with_capacity(
driver: VirtIOSocket<H, T, RX_BUFFER_SIZE>,
per_connection_buffer_capacity: usize,
) -> Self {
Self {
driver,
connections: Vec::new(),
listening_ports: Vec::new(),
per_connection_buffer_capacity,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Are there any requirements on per_connection_buffer_capacity similar to RX_BUFFER_SIZE?

If so, would it make sense to validate the buffer size here or would an invalid buffer fail a check elsewhere?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As far as I can tell the only requirement is that it fit in a u32, so I've changed the type to that.

}
}

Expand Down Expand Up @@ -106,7 +126,8 @@ impl<H: Hal, T: Transport> VsockConnectionManager<H, T> {
return Err(SocketError::ConnectionExists.into());
}

let new_connection = Connection::new(destination, src_port);
let new_connection =
Connection::new(destination, src_port, self.per_connection_buffer_capacity);

self.driver.connect(&new_connection.info)?;
debug!("Connection requested: {:?}", new_connection.info);
Expand All @@ -125,6 +146,7 @@ impl<H: Hal, T: Transport> VsockConnectionManager<H, T> {
pub fn poll(&mut self) -> Result<Option<VsockEvent>> {
let guest_cid = self.driver.guest_cid();
let connections = &mut self.connections;
let per_connection_buffer_capacity = self.per_connection_buffer_capacity;

let result = self.driver.poll(|event, body| {
let connection = get_connection_for_event(connections, &event, guest_cid);
Expand All @@ -140,7 +162,11 @@ impl<H: Hal, T: Transport> VsockConnectionManager<H, T> {
}
// Add the new connection to our list, at least for now. It will be removed again
// below if we weren't listening on the port.
connections.push(Connection::new(event.source, event.destination.port));
connections.push(Connection::new(
event.source,
event.destination.port,
per_connection_buffer_capacity,
));
connections.last_mut().unwrap()
} else {
return Ok(None);
Expand Down
4 changes: 4 additions & 0 deletions src/device/socket/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,3 +20,7 @@ pub use error::SocketError;
pub use protocol::{VsockAddr, VMADDR_CID_HOST};
#[cfg(feature = "alloc")]
pub use vsock::{DisconnectReason, VirtIOSocket, VsockEvent, VsockEventType};

/// The size in bytes of each buffer used in the RX virtqueue. This must be bigger than
/// `size_of::<VirtioVsockHdr>()`.
const DEFAULT_RX_BUFFER_SIZE: usize = 512;
18 changes: 12 additions & 6 deletions src/device/socket/vsock.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ use super::error::SocketError;
use super::protocol::{
Feature, StreamShutdown, VirtioVsockConfig, VirtioVsockHdr, VirtioVsockOp, VsockAddr,
};
use super::DEFAULT_RX_BUFFER_SIZE;
use crate::hal::Hal;
use crate::queue::VirtQueue;
use crate::transport::Transport;
Expand All @@ -23,9 +24,6 @@ const EVENT_QUEUE_IDX: u16 = 2;
pub(crate) const QUEUE_SIZE: usize = 8;
const SUPPORTED_FEATURES: Feature = Feature::RING_EVENT_IDX;

/// The size in bytes of each buffer used in the RX virtqueue. This must be bigger than size_of::<VirtioVsockHdr>().
const RX_BUFFER_SIZE: usize = 512;

#[derive(Clone, Debug, Default, PartialEq, Eq)]
pub struct ConnectionInfo {
pub dst: VsockAddr,
Expand Down Expand Up @@ -212,7 +210,11 @@ pub enum VsockEventType {
///
/// You probably want to use [`VsockConnectionManager`](super::VsockConnectionManager) rather than
/// using this directly.
pub struct VirtIOSocket<H: Hal, T: Transport> {
///
/// `RX_BUFFER_SIZE` is the size in bytes of each buffer used in the RX virtqueue. This must be
/// bigger than `size_of::<VirtioVsockHdr>()`.
pub struct VirtIOSocket<H: Hal, T: Transport, const RX_BUFFER_SIZE: usize = DEFAULT_RX_BUFFER_SIZE>
{
transport: T,
/// Virtqueue to receive packets.
rx: VirtQueue<H, { QUEUE_SIZE }>,
Expand All @@ -237,7 +239,9 @@ unsafe impl<H: Hal, T: Transport + Sync> Sync for VirtIOSocket<H, T> where
{
}

impl<H: Hal, T: Transport> Drop for VirtIOSocket<H, T> {
impl<H: Hal, T: Transport, const RX_BUFFER_SIZE: usize> Drop
for VirtIOSocket<H, T, RX_BUFFER_SIZE>
{
fn drop(&mut self) {
// Clear any pointers pointing to DMA regions, so the device doesn't try to access them
// after they have been freed.
Expand All @@ -253,9 +257,11 @@ impl<H: Hal, T: Transport> Drop for VirtIOSocket<H, T> {
}
}

impl<H: Hal, T: Transport> VirtIOSocket<H, T> {
impl<H: Hal, T: Transport, const RX_BUFFER_SIZE: usize> VirtIOSocket<H, T, RX_BUFFER_SIZE> {
/// Create a new VirtIO Vsock driver.
pub fn new(mut transport: T) -> Result<Self> {
assert!(RX_BUFFER_SIZE > size_of::<VirtioVsockHdr>());

let negotiated_features = transport.begin_init(SUPPORTED_FEATURES);

let config = transport.config_space::<VirtioVsockConfig>()?;
Expand Down
Loading