mirror of
https://chromium.googlesource.com/crosvm/crosvm
synced 2025-02-06 10:32:10 +00:00
d49bcdb081
Replaced all usage of MsgSocket with Tube. BUG=b:176847362 TEST=run_tests Change-Id: Ie647f79926bc0414c125a776eafe7f60c071bab2 Reviewed-on: https://chromium-review.googlesource.com/c/chromiumos/platform/crosvm/+/2710324 Auto-Submit: Zach Reizner <zachr@chromium.org> Tested-by: kokoro <noreply+kokoro@google.com> Commit-Queue: Zach Reizner <zachr@chromium.org> Reviewed-by: Dylan Reid <dgreid@chromium.org>
481 lines
15 KiB
Rust
481 lines
15 KiB
Rust
// Copyright 2017 The Chromium OS Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
//! Handles routing to devices in an address space.
|
|
|
|
use std::cmp::{Ord, Ordering, PartialEq, PartialOrd};
|
|
use std::collections::btree_map::BTreeMap;
|
|
use std::fmt::{self, Display};
|
|
use std::result;
|
|
use std::sync::Arc;
|
|
|
|
use serde::{Deserialize, Serialize};
|
|
use sync::Mutex;
|
|
|
|
/// Information about how a device was accessed.
|
|
#[derive(Copy, Clone, Eq, PartialEq, Debug, Serialize, Deserialize)]
|
|
pub struct BusAccessInfo {
|
|
/// Offset from base address that the device was accessed at.
|
|
pub offset: u64,
|
|
/// Absolute address of the device's access in its address space.
|
|
pub address: u64,
|
|
/// ID of the entity requesting a device access, usually the VCPU id.
|
|
pub id: usize,
|
|
}
|
|
|
|
// Implement `Display` for `MinMax`.
|
|
impl std::fmt::Display for BusAccessInfo {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(f, "{:?}", self)
|
|
}
|
|
}
|
|
|
|
/// Result of a write to a device's PCI configuration space.
|
|
/// This value represents the state change(s) that occurred due to the write.
|
|
/// Each member of this structure may be `None` if no change occurred, or `Some(new_value)` to
|
|
/// indicate a state change.
|
|
#[derive(Copy, Clone, Debug, Default, PartialEq)]
|
|
pub struct ConfigWriteResult {
|
|
/// New state of the memory bus for this PCI device:
|
|
/// - `None`: no change in state.
|
|
/// - `Some(true)`: memory decode enabled; device should respond to memory accesses.
|
|
/// - `Some(false)`: memory decode disabled; device should not respond to memory accesses.
|
|
pub mem_bus_new_state: Option<bool>,
|
|
|
|
/// New state of the I/O bus for this PCI device:
|
|
/// - `None`: no change in state.
|
|
/// - `Some(true)`: I/O decode enabled; device should respond to I/O accesses.
|
|
/// - `Some(false)`: I/O decode disabled; device should not respond to I/O accesses.
|
|
pub io_bus_new_state: Option<bool>,
|
|
}
|
|
|
|
/// Trait for devices that respond to reads or writes in an arbitrary address space.
|
|
///
|
|
/// The device does not care where it exists in address space as each method is only given an offset
|
|
/// into its allocated portion of address space.
|
|
#[allow(unused_variables)]
|
|
pub trait BusDevice: Send {
|
|
/// Returns a label suitable for debug output.
|
|
fn debug_label(&self) -> String;
|
|
/// Reads at `offset` from this device
|
|
fn read(&mut self, offset: BusAccessInfo, data: &mut [u8]) {}
|
|
/// Writes at `offset` into this device
|
|
fn write(&mut self, offset: BusAccessInfo, data: &[u8]) {}
|
|
/// Sets a register in the configuration space. Only used by PCI.
|
|
/// * `reg_idx` - The index of the config register to modify.
|
|
/// * `offset` - Offset in to the register.
|
|
fn config_register_write(
|
|
&mut self,
|
|
reg_idx: usize,
|
|
offset: u64,
|
|
data: &[u8],
|
|
) -> ConfigWriteResult {
|
|
ConfigWriteResult {
|
|
..Default::default()
|
|
}
|
|
}
|
|
/// Gets a register from the configuration space. Only used by PCI.
|
|
/// * `reg_idx` - The index of the config register to read.
|
|
fn config_register_read(&self, reg_idx: usize) -> u32 {
|
|
0
|
|
}
|
|
/// Invoked when the device is sandboxed.
|
|
fn on_sandboxed(&mut self) {}
|
|
}
|
|
|
|
pub trait BusDeviceSync: BusDevice + Sync {
|
|
fn read(&self, offset: BusAccessInfo, data: &mut [u8]);
|
|
fn write(&self, offset: BusAccessInfo, data: &[u8]);
|
|
}
|
|
|
|
pub trait BusResumeDevice: Send {
|
|
/// notify the devices which are invoked
|
|
/// before the VM resumes form suspend.
|
|
fn resume_imminent(&mut self) {}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
pub enum Error {
|
|
/// The insertion failed because the new device overlapped with an old device.
|
|
Overlap,
|
|
}
|
|
|
|
impl Display for Error {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
use self::Error::*;
|
|
|
|
match self {
|
|
Overlap => write!(f, "new device overlaps with an old device"),
|
|
}
|
|
}
|
|
}
|
|
|
|
pub type Result<T> = result::Result<T, Error>;
|
|
|
|
/// Holds a base and length representing the address space occupied by a `BusDevice`.
|
|
///
|
|
/// * base - The address at which the range start.
|
|
/// * len - The length of the range in bytes.
|
|
#[derive(Debug, Copy, Clone)]
|
|
pub struct BusRange {
|
|
pub base: u64,
|
|
pub len: u64,
|
|
}
|
|
|
|
impl BusRange {
|
|
/// Returns true if `addr` is within the range.
|
|
pub fn contains(&self, addr: u64) -> bool {
|
|
self.base <= addr && addr < self.base + self.len
|
|
}
|
|
|
|
/// Returns true if there is overlap with the given range.
|
|
pub fn overlaps(&self, base: u64, len: u64) -> bool {
|
|
self.base < (base + len) && base < self.base + self.len
|
|
}
|
|
}
|
|
|
|
impl Eq for BusRange {}
|
|
|
|
impl PartialEq for BusRange {
|
|
fn eq(&self, other: &BusRange) -> bool {
|
|
self.base == other.base
|
|
}
|
|
}
|
|
|
|
impl Ord for BusRange {
|
|
fn cmp(&self, other: &BusRange) -> Ordering {
|
|
self.base.cmp(&other.base)
|
|
}
|
|
}
|
|
|
|
impl PartialOrd for BusRange {
|
|
fn partial_cmp(&self, other: &BusRange) -> Option<Ordering> {
|
|
self.base.partial_cmp(&other.base)
|
|
}
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
enum BusDeviceEntry {
|
|
OuterSync(Arc<Mutex<dyn BusDevice>>),
|
|
InnerSync(Arc<dyn BusDeviceSync>),
|
|
}
|
|
|
|
/// A device container for routing reads and writes over some address space.
|
|
///
|
|
/// This doesn't have any restrictions on what kind of device or address space this applies to. The
|
|
/// only restriction is that no two devices can overlap in this address space.
|
|
///
|
|
/// the 'resume_notify_devices' contains the devices which requires to be notified before the system
|
|
/// resume back from S3 suspended state.
|
|
#[derive(Clone)]
|
|
pub struct Bus {
|
|
devices: BTreeMap<BusRange, BusDeviceEntry>,
|
|
resume_notify_devices: Vec<Arc<Mutex<dyn BusResumeDevice>>>,
|
|
access_id: usize,
|
|
}
|
|
|
|
impl Bus {
|
|
/// Constructs an a bus with an empty address space.
|
|
pub fn new() -> Bus {
|
|
Bus {
|
|
devices: BTreeMap::new(),
|
|
resume_notify_devices: Vec::new(),
|
|
access_id: 0,
|
|
}
|
|
}
|
|
|
|
/// Sets the id that will be used for BusAccessInfo.
|
|
pub fn set_access_id(&mut self, id: usize) {
|
|
self.access_id = id;
|
|
}
|
|
|
|
fn first_before(&self, addr: u64) -> Option<(BusRange, &BusDeviceEntry)> {
|
|
let (range, dev) = self
|
|
.devices
|
|
.range(..=BusRange { base: addr, len: 1 })
|
|
.rev()
|
|
.next()?;
|
|
Some((*range, dev))
|
|
}
|
|
|
|
fn get_device(&self, addr: u64) -> Option<(u64, u64, &BusDeviceEntry)> {
|
|
if let Some((range, dev)) = self.first_before(addr) {
|
|
let offset = addr - range.base;
|
|
if offset < range.len {
|
|
return Some((offset, addr, dev));
|
|
}
|
|
}
|
|
None
|
|
}
|
|
|
|
/// Puts the given device at the given address space.
|
|
pub fn insert(&mut self, device: Arc<Mutex<dyn BusDevice>>, base: u64, len: u64) -> Result<()> {
|
|
if len == 0 {
|
|
return Err(Error::Overlap);
|
|
}
|
|
|
|
// Reject all cases where the new device's range overlaps with an existing device.
|
|
if self
|
|
.devices
|
|
.iter()
|
|
.any(|(range, _dev)| range.overlaps(base, len))
|
|
{
|
|
return Err(Error::Overlap);
|
|
}
|
|
|
|
if self
|
|
.devices
|
|
.insert(BusRange { base, len }, BusDeviceEntry::OuterSync(device))
|
|
.is_some()
|
|
{
|
|
return Err(Error::Overlap);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Puts the given device that implements BusDeviceSync at the given address space. Devices
|
|
/// that implement BusDeviceSync manage thread safety internally, and thus can be written to
|
|
/// by multiple threads simultaneously.
|
|
pub fn insert_sync(
|
|
&mut self,
|
|
device: Arc<dyn BusDeviceSync>,
|
|
base: u64,
|
|
len: u64,
|
|
) -> Result<()> {
|
|
if len == 0 {
|
|
return Err(Error::Overlap);
|
|
}
|
|
|
|
// Reject all cases where the new device's range overlaps with an existing device.
|
|
if self
|
|
.devices
|
|
.iter()
|
|
.any(|(range, _dev)| range.overlaps(base, len))
|
|
{
|
|
return Err(Error::Overlap);
|
|
}
|
|
|
|
if self
|
|
.devices
|
|
.insert(BusRange { base, len }, BusDeviceEntry::InnerSync(device))
|
|
.is_some()
|
|
{
|
|
return Err(Error::Overlap);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Reads data from the device that owns the range containing `addr` and puts it into `data`.
|
|
///
|
|
/// Returns true on success, otherwise `data` is untouched.
|
|
pub fn read(&self, addr: u64, data: &mut [u8]) -> bool {
|
|
if let Some((offset, address, dev)) = self.get_device(addr) {
|
|
let io = BusAccessInfo {
|
|
address,
|
|
offset,
|
|
id: self.access_id,
|
|
};
|
|
match dev {
|
|
BusDeviceEntry::OuterSync(dev) => dev.lock().read(io, data),
|
|
BusDeviceEntry::InnerSync(dev) => dev.read(io, data),
|
|
}
|
|
true
|
|
} else {
|
|
false
|
|
}
|
|
}
|
|
|
|
/// Writes `data` to the device that owns the range containing `addr`.
|
|
///
|
|
/// Returns true on success, otherwise `data` is untouched.
|
|
pub fn write(&self, addr: u64, data: &[u8]) -> bool {
|
|
if let Some((offset, address, dev)) = self.get_device(addr) {
|
|
let io = BusAccessInfo {
|
|
address,
|
|
offset,
|
|
id: self.access_id,
|
|
};
|
|
match dev {
|
|
BusDeviceEntry::OuterSync(dev) => dev.lock().write(io, data),
|
|
BusDeviceEntry::InnerSync(dev) => dev.write(io, data),
|
|
}
|
|
true
|
|
} else {
|
|
false
|
|
}
|
|
}
|
|
|
|
/// Register `device` for notifications of VM resume from suspend.
|
|
pub fn notify_on_resume(&mut self, device: Arc<Mutex<dyn BusResumeDevice>>) {
|
|
self.resume_notify_devices.push(device);
|
|
}
|
|
|
|
/// Call `notify_resume` to notify the device that suspend resume is imminent.
|
|
pub fn notify_resume(&mut self) {
|
|
let devices = self.resume_notify_devices.clone();
|
|
for dev in devices {
|
|
dev.lock().resume_imminent();
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
|
|
struct DummyDevice;
|
|
impl BusDevice for DummyDevice {
|
|
fn debug_label(&self) -> String {
|
|
"dummy device".to_owned()
|
|
}
|
|
}
|
|
|
|
struct ConstantDevice {
|
|
uses_full_addr: bool,
|
|
}
|
|
|
|
impl BusDevice for ConstantDevice {
|
|
fn debug_label(&self) -> String {
|
|
"constant device".to_owned()
|
|
}
|
|
|
|
fn read(&mut self, info: BusAccessInfo, data: &mut [u8]) {
|
|
let addr = if self.uses_full_addr {
|
|
info.address
|
|
} else {
|
|
info.offset
|
|
};
|
|
for (i, v) in data.iter_mut().enumerate() {
|
|
*v = (addr as u8) + (i as u8);
|
|
}
|
|
}
|
|
|
|
fn write(&mut self, info: BusAccessInfo, data: &[u8]) {
|
|
let addr = if self.uses_full_addr {
|
|
info.address
|
|
} else {
|
|
info.offset
|
|
};
|
|
for (i, v) in data.iter().enumerate() {
|
|
assert_eq!(*v, (addr as u8) + (i as u8))
|
|
}
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn bus_insert() {
|
|
let mut bus = Bus::new();
|
|
let dummy = Arc::new(Mutex::new(DummyDevice));
|
|
assert!(bus.insert(dummy.clone(), 0x10, 0).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x10, 0x10).is_ok());
|
|
assert!(bus.insert(dummy.clone(), 0x0f, 0x10).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x10, 0x10).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x10, 0x15).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x12, 0x15).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x12, 0x01).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x0, 0x20).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x20, 0x05).is_ok());
|
|
assert!(bus.insert(dummy.clone(), 0x25, 0x05).is_ok());
|
|
assert!(bus.insert(dummy.clone(), 0x0, 0x10).is_ok());
|
|
}
|
|
|
|
#[test]
|
|
fn bus_insert_full_addr() {
|
|
let mut bus = Bus::new();
|
|
let dummy = Arc::new(Mutex::new(DummyDevice));
|
|
assert!(bus.insert(dummy.clone(), 0x10, 0).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x10, 0x10).is_ok());
|
|
assert!(bus.insert(dummy.clone(), 0x0f, 0x10).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x10, 0x10).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x10, 0x15).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x12, 0x15).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x12, 0x01).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x0, 0x20).is_err());
|
|
assert!(bus.insert(dummy.clone(), 0x20, 0x05).is_ok());
|
|
assert!(bus.insert(dummy.clone(), 0x25, 0x05).is_ok());
|
|
assert!(bus.insert(dummy.clone(), 0x0, 0x10).is_ok());
|
|
}
|
|
|
|
#[test]
|
|
fn bus_read_write() {
|
|
let mut bus = Bus::new();
|
|
let dummy = Arc::new(Mutex::new(DummyDevice));
|
|
assert!(bus.insert(dummy.clone(), 0x10, 0x10).is_ok());
|
|
assert!(bus.read(0x10, &mut [0, 0, 0, 0]));
|
|
assert!(bus.write(0x10, &[0, 0, 0, 0]));
|
|
assert!(bus.read(0x11, &mut [0, 0, 0, 0]));
|
|
assert!(bus.write(0x11, &[0, 0, 0, 0]));
|
|
assert!(bus.read(0x16, &mut [0, 0, 0, 0]));
|
|
assert!(bus.write(0x16, &[0, 0, 0, 0]));
|
|
assert!(!bus.read(0x20, &mut [0, 0, 0, 0]));
|
|
assert!(!bus.write(0x20, &mut [0, 0, 0, 0]));
|
|
assert!(!bus.read(0x06, &mut [0, 0, 0, 0]));
|
|
assert!(!bus.write(0x06, &mut [0, 0, 0, 0]));
|
|
}
|
|
|
|
#[test]
|
|
fn bus_read_write_values() {
|
|
let mut bus = Bus::new();
|
|
let dummy = Arc::new(Mutex::new(ConstantDevice {
|
|
uses_full_addr: false,
|
|
}));
|
|
assert!(bus.insert(dummy.clone(), 0x10, 0x10).is_ok());
|
|
|
|
let mut values = [0, 1, 2, 3];
|
|
assert!(bus.read(0x10, &mut values));
|
|
assert_eq!(values, [0, 1, 2, 3]);
|
|
assert!(bus.write(0x10, &values));
|
|
assert!(bus.read(0x15, &mut values));
|
|
assert_eq!(values, [5, 6, 7, 8]);
|
|
assert!(bus.write(0x15, &values));
|
|
}
|
|
|
|
#[test]
|
|
fn bus_read_write_full_addr_values() {
|
|
let mut bus = Bus::new();
|
|
let dummy = Arc::new(Mutex::new(ConstantDevice {
|
|
uses_full_addr: true,
|
|
}));
|
|
assert!(bus.insert(dummy.clone(), 0x10, 0x10).is_ok());
|
|
|
|
let mut values = [0u8; 4];
|
|
assert!(bus.read(0x10, &mut values));
|
|
assert_eq!(values, [0x10, 0x11, 0x12, 0x13]);
|
|
assert!(bus.write(0x10, &values));
|
|
assert!(bus.read(0x15, &mut values));
|
|
assert_eq!(values, [0x15, 0x16, 0x17, 0x18]);
|
|
assert!(bus.write(0x15, &values));
|
|
}
|
|
|
|
#[test]
|
|
fn bus_range_contains() {
|
|
let a = BusRange {
|
|
base: 0x1000,
|
|
len: 0x400,
|
|
};
|
|
assert!(a.contains(0x1000));
|
|
assert!(a.contains(0x13ff));
|
|
assert!(!a.contains(0xfff));
|
|
assert!(!a.contains(0x1400));
|
|
assert!(a.contains(0x1200));
|
|
}
|
|
|
|
#[test]
|
|
fn bus_range_overlap() {
|
|
let a = BusRange {
|
|
base: 0x1000,
|
|
len: 0x400,
|
|
};
|
|
assert!(a.overlaps(0x1000, 0x400));
|
|
assert!(a.overlaps(0xf00, 0x400));
|
|
assert!(a.overlaps(0x1000, 0x01));
|
|
assert!(a.overlaps(0xfff, 0x02));
|
|
assert!(a.overlaps(0x1100, 0x100));
|
|
assert!(a.overlaps(0x13ff, 0x100));
|
|
assert!(!a.overlaps(0x1400, 0x100));
|
|
assert!(!a.overlaps(0xf00, 0x100));
|
|
}
|
|
}
|