vm-virtio: Add support for indirect descriptors

Indirect descriptors is a virtio feature that allows the driver to
store a table of descriptors anywhere in memory, pointing to it from a
virtqueue ring's descriptor with a particular flag.

We can't seamlessly transition from an iterator over a conventional
descriptor chain to an indirect chain, so Queue users need to
explicitly support this feature by calling Queue::is_indirect() and
Queue::new_from_indirect().

Signed-off-by: Sergio Lopez <slp@redhat.com>
This commit is contained in:
Sergio Lopez 2020-02-21 11:34:51 +01:00 committed by Rob Bradford
parent d7b0b9842d
commit 42937c9754
3 changed files with 116 additions and 12 deletions

View File

@ -27,8 +27,8 @@ vmm-sys-util = ">=0.3.1"
[dependencies.vhost_rs] [dependencies.vhost_rs]
path = "../vhost_rs" path = "../vhost_rs"
features = ["vhost-user-master"] features = ["vhost-user-master", "vhost-user-slave"]
[dependencies.vm-memory] [dependencies.vm-memory]
git = "https://github.com/rust-vmm/vm-memory" git = "https://github.com/rust-vmm/vm-memory"
features = ["backend-mmap"] features = ["backend-mmap", "backend-atomic"]

View File

@ -32,7 +32,7 @@ mod iommu;
pub mod net; pub mod net;
pub mod net_util; pub mod net_util;
mod pmem; mod pmem;
mod queue; pub mod queue;
mod rng; mod rng;
pub mod vsock; pub mod vsock;

View File

@ -9,6 +9,8 @@
// SPDX-License-Identifier: Apache-2.0 AND BSD-3-Clause // SPDX-License-Identifier: Apache-2.0 AND BSD-3-Clause
use std::cmp::min; use std::cmp::min;
use std::convert::TryInto;
use std::fmt::{self, Display};
use std::num::Wrapping; use std::num::Wrapping;
use std::sync::atomic::{fence, Ordering}; use std::sync::atomic::{fence, Ordering};
use std::sync::Arc; use std::sync::Arc;
@ -20,6 +22,26 @@ use vm_memory::{
pub(super) const VIRTQ_DESC_F_NEXT: u16 = 0x1; pub(super) const VIRTQ_DESC_F_NEXT: u16 = 0x1;
pub(super) const VIRTQ_DESC_F_WRITE: u16 = 0x2; pub(super) const VIRTQ_DESC_F_WRITE: u16 = 0x2;
pub(super) const VIRTQ_DESC_F_INDIRECT: u16 = 0x4;
#[derive(Debug)]
pub enum Error {
GuestMemoryError,
InvalidIndirectDescriptor,
InvalidChain,
}
impl Display for Error {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
use self::Error::*;
match self {
GuestMemoryError => write!(f, "error accessing guest memory"),
InvalidChain => write!(f, "invalid descriptor chain"),
InvalidIndirectDescriptor => write!(f, "invalid indirect descriptor"),
}
}
}
// GuestMemoryMmap::read_obj() will be used to fetch the descriptor, // GuestMemoryMmap::read_obj() will be used to fetch the descriptor,
// which has an explicit constraint that the entire descriptor doesn't // which has an explicit constraint that the entire descriptor doesn't
@ -76,7 +98,7 @@ unsafe impl ByteValued for Descriptor {}
#[derive(Clone)] #[derive(Clone)]
pub struct DescriptorChain<'a> { pub struct DescriptorChain<'a> {
desc_table: GuestAddress, desc_table: GuestAddress,
queue_size: u16, table_size: u16,
ttl: u16, // used to prevent infinite chain cycles ttl: u16, // used to prevent infinite chain cycles
iommu_mapping_cb: Option<Arc<VirtioIommuRemapping>>, iommu_mapping_cb: Option<Arc<VirtioIommuRemapping>>,
@ -104,11 +126,11 @@ impl<'a> DescriptorChain<'a> {
pub fn checked_new( pub fn checked_new(
mem: &GuestMemoryMmap, mem: &GuestMemoryMmap,
desc_table: GuestAddress, desc_table: GuestAddress,
queue_size: u16, table_size: u16,
index: u16, index: u16,
iommu_mapping_cb: Option<Arc<VirtioIommuRemapping>>, iommu_mapping_cb: Option<Arc<VirtioIommuRemapping>>,
) -> Option<DescriptorChain> { ) -> Option<DescriptorChain> {
if index >= queue_size { if index >= table_size {
return None; return None;
} }
@ -138,8 +160,8 @@ impl<'a> DescriptorChain<'a> {
let chain = DescriptorChain { let chain = DescriptorChain {
mem, mem,
desc_table, desc_table,
queue_size, table_size,
ttl: queue_size, ttl: table_size,
index, index,
addr: GuestAddress(desc_addr), addr: GuestAddress(desc_addr),
len: desc.len, len: desc.len,
@ -155,12 +177,59 @@ impl<'a> DescriptorChain<'a> {
} }
} }
pub fn new_from_indirect(&self) -> Result<DescriptorChain, Error> {
if !self.is_indirect() {
return Err(Error::InvalidIndirectDescriptor);
}
let desc_head = self.addr;
self.mem
.checked_offset(desc_head, 16)
.ok_or(Error::GuestMemoryError)?;
// These reads can't fail unless Guest memory is hopelessly broken.
let desc = match self.mem.read_obj::<Descriptor>(desc_head) {
Ok(ret) => ret,
Err(_) => return Err(Error::GuestMemoryError),
};
// Translate address if necessary
let (desc_addr, iommu_mapping_cb) =
if let Some(iommu_mapping_cb) = self.iommu_mapping_cb.clone() {
(
(iommu_mapping_cb)(desc.addr).unwrap(),
Some(iommu_mapping_cb),
)
} else {
(desc.addr, None)
};
let chain = DescriptorChain {
mem: self.mem,
desc_table: self.addr,
table_size: (self.len / 16).try_into().unwrap(),
ttl: (self.len / 16).try_into().unwrap(),
index: 0,
addr: GuestAddress(desc_addr),
len: desc.len,
flags: desc.flags,
next: desc.next,
iommu_mapping_cb,
};
if !chain.is_valid() {
return Err(Error::InvalidChain);
}
Ok(chain)
}
fn is_valid(&self) -> bool { fn is_valid(&self) -> bool {
!(self !(self
.mem .mem
.checked_offset(self.addr, self.len as usize) .checked_offset(self.addr, self.len as usize)
.is_none() .is_none()
|| (self.has_next() && self.next >= self.queue_size)) || (self.has_next() && self.next >= self.table_size))
} }
/// Gets if this descriptor chain has another descriptor chain linked after it. /// Gets if this descriptor chain has another descriptor chain linked after it.
@ -176,6 +245,10 @@ impl<'a> DescriptorChain<'a> {
self.flags & VIRTQ_DESC_F_WRITE != 0 self.flags & VIRTQ_DESC_F_WRITE != 0
} }
pub fn is_indirect(&self) -> bool {
self.flags & VIRTQ_DESC_F_INDIRECT != 0
}
/// Gets the next descriptor in this descriptor chain, if there is one. /// Gets the next descriptor in this descriptor chain, if there is one.
/// ///
/// Note that this is distinct from the next descriptor chain returned by `AvailIter`, which is /// Note that this is distinct from the next descriptor chain returned by `AvailIter`, which is
@ -185,7 +258,7 @@ impl<'a> DescriptorChain<'a> {
DescriptorChain::checked_new( DescriptorChain::checked_new(
self.mem, self.mem,
self.desc_table, self.desc_table,
self.queue_size, self.table_size,
self.next, self.next,
self.iommu_mapping_cb.clone(), self.iommu_mapping_cb.clone(),
) )
@ -825,8 +898,8 @@ pub(crate) mod tests {
assert_eq!(c.mem as *const GuestMemoryMmap, m as *const GuestMemoryMmap); assert_eq!(c.mem as *const GuestMemoryMmap, m as *const GuestMemoryMmap);
assert_eq!(c.desc_table, vq.start()); assert_eq!(c.desc_table, vq.start());
assert_eq!(c.queue_size, 16); assert_eq!(c.table_size, 16);
assert_eq!(c.ttl, c.queue_size); assert_eq!(c.ttl, c.table_size);
assert_eq!(c.index, 0); assert_eq!(c.index, 0);
assert_eq!(c.addr, GuestAddress(0x1000)); assert_eq!(c.addr, GuestAddress(0x1000));
assert_eq!(c.len, 0x1000); assert_eq!(c.len, 0x1000);
@ -837,6 +910,37 @@ pub(crate) mod tests {
} }
} }
#[test]
fn test_new_from_descriptor_chain() {
let m = &GuestMemoryMmap::from_ranges(&[(GuestAddress(0), 0x10000)]).unwrap();
let vq = VirtQueue::new(GuestAddress(0), m, 16);
// create a chain with a descriptor pointing to an indirect table
vq.dtable[0].addr.set(0x1000);
vq.dtable[0].len.set(0x1000);
vq.dtable[0].next.set(0);
vq.dtable[0].flags.set(VIRTQ_DESC_F_INDIRECT);
let c = DescriptorChain::checked_new(m, vq.start(), 16, 0, None).unwrap();
assert!(c.is_indirect());
// create an indirect table with 4 chained descriptors
let mut indirect_table = Vec::with_capacity(4 as usize);
for j in 0..4 {
let desc = VirtqDesc::new(GuestAddress(0x1000 + (j * 16)), m);
desc.set(0x1000, 0x1000, VIRTQ_DESC_F_NEXT, (j + 1) as u16);
indirect_table.push(desc);
}
// try to iterate through the indirect table descriptors
let mut i = c.new_from_indirect().unwrap();
for j in 0..4 {
assert_eq!(i.flags, VIRTQ_DESC_F_NEXT);
assert_eq!(i.next, j + 1);
i = i.next_descriptor().unwrap();
}
}
#[test] #[test]
fn test_queue_and_iterator() { fn test_queue_and_iterator() {
let m = &GuestMemoryMmap::from_ranges(&[(GuestAddress(0), 0x10000)]).unwrap(); let m = &GuestMemoryMmap::from_ranges(&[(GuestAddress(0), 0x10000)]).unwrap();