mirror of
https://github.com/cloud-hypervisor/cloud-hypervisor.git
synced 2025-01-08 22:05:20 +00:00
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:
parent
d7b0b9842d
commit
42937c9754
@ -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"]
|
||||||
|
@ -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;
|
||||||
|
|
||||||
|
@ -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();
|
||||||
|
Loading…
Reference in New Issue
Block a user