hypervisor: Refactor create_passthrough_device() for generic type

Changed the return type of create_passthrough_device() to generic type
hypervisor::Device.

Signed-off-by: Michael Zhao <michael.zhao@arm.com>
This commit is contained in:
Michael Zhao 2020-07-21 17:56:00 +08:00 committed by Samuel Ortiz
parent 6c8749adf2
commit ddf1b76906
4 changed files with 32 additions and 22 deletions

View File

@ -10,6 +10,7 @@
// //
use crate::DeviceAttr; use crate::DeviceAttr;
use std::os::unix::io::AsRawFd;
use thiserror::Error; use thiserror::Error;
#[derive(Error, Debug)] #[derive(Error, Debug)]
@ -33,7 +34,7 @@ pub type Result<T> = std::result::Result<T, HypervisorDeviceError>;
/// ///
/// This crate provides a hypervisor-agnostic interfaces for device /// This crate provides a hypervisor-agnostic interfaces for device
/// ///
pub trait Device: Send + Sync { pub trait Device: Send + Sync + AsRawFd {
/// Set device attribute. /// Set device attribute.
fn set_device_attr(&self, attr: &DeviceAttr) -> Result<()>; fn set_device_attr(&self, attr: &DeviceAttr) -> Result<()>;
} }

View File

@ -9,6 +9,7 @@
// //
use kvm_ioctls::{NoDatamatch, VcpuFd, VmFd}; use kvm_ioctls::{NoDatamatch, VcpuFd, VmFd};
use std::os::unix::io::{AsRawFd, RawFd};
use std::result; use std::result;
use std::sync::Arc; use std::sync::Arc;
#[cfg(target_arch = "x86_64")] #[cfg(target_arch = "x86_64")]
@ -303,15 +304,14 @@ impl vm::Vm for KvmVm {
self.fd.check_extension(c) self.fd.check_extension(c)
} }
/// Create a device that is used for passthrough /// Create a device that is used for passthrough
fn create_passthrough_device(&self) -> vm::Result<DeviceFd> { fn create_passthrough_device(&self) -> vm::Result<Arc<dyn device::Device>> {
let mut vfio_dev = kvm_create_device { let mut vfio_dev = kvm_create_device {
type_: kvm_device_type_KVM_DEV_TYPE_VFIO, type_: kvm_device_type_KVM_DEV_TYPE_VFIO,
fd: 0, fd: 0,
flags: 0, flags: 0,
}; };
self.fd self.create_device(&mut vfio_dev)
.create_device(&mut vfio_dev)
.map_err(|e| vm::HypervisorVmError::CreatePassthroughDevice(e.into())) .map_err(|e| vm::HypervisorVmError::CreatePassthroughDevice(e.into()))
} }
} }
@ -869,3 +869,9 @@ impl device::Device for KvmDevice {
.map_err(|e| device::HypervisorDeviceError::SetDeviceAttribute(e.into())) .map_err(|e| device::HypervisorDeviceError::SetDeviceAttribute(e.into()))
} }
} }
impl AsRawFd for KvmDevice {
fn as_raw_fd(&self) -> RawFd {
self.fd.as_raw_fd()
}
}

View File

@ -14,7 +14,7 @@ use crate::cpu::Vcpu;
use crate::device::Device; use crate::device::Device;
#[cfg(target_arch = "x86_64")] #[cfg(target_arch = "x86_64")]
use crate::ClockData; use crate::ClockData;
use crate::{CreateDevice, DeviceFd, IoEventAddress, IrqRoutingEntry, MemoryRegion}; use crate::{CreateDevice, IoEventAddress, IrqRoutingEntry, MemoryRegion};
use kvm_ioctls::Cap; use kvm_ioctls::Cap;
use std::sync::Arc; use std::sync::Arc;
use thiserror::Error; use thiserror::Error;
@ -178,5 +178,5 @@ pub trait Vm: Send + Sync {
/// Checks if a particular `Cap` is available. /// Checks if a particular `Cap` is available.
fn check_extension(&self, c: Cap) -> bool; fn check_extension(&self, c: Cap) -> bool;
/// Create a device that is used for passthrough /// Create a device that is used for passthrough
fn create_passthrough_device(&self) -> Result<DeviceFd>; fn create_passthrough_device(&self) -> Result<Arc<dyn Device>>;
} }

View File

@ -57,6 +57,8 @@ use std::fs::{File, OpenOptions};
use std::io::{self, sink, stdout, Seek, SeekFrom}; use std::io::{self, sink, stdout, Seek, SeekFrom};
use std::num::Wrapping; use std::num::Wrapping;
use std::os::unix::fs::OpenOptionsExt; use std::os::unix::fs::OpenOptionsExt;
#[cfg(feature = "pci_support")]
use std::os::unix::io::FromRawFd;
use std::path::PathBuf; use std::path::PathBuf;
use std::result; use std::result;
use std::sync::{Arc, Mutex}; use std::sync::{Arc, Mutex};
@ -745,7 +747,7 @@ pub struct DeviceManager {
// Passthrough device handle // Passthrough device handle
#[cfg(feature = "pci_support")] #[cfg(feature = "pci_support")]
passthrough_device: Option<Arc<DeviceFd>>, passthrough_device: Option<Arc<dyn hypervisor::Device>>,
// Paravirtualized IOMMU // Paravirtualized IOMMU
#[cfg(feature = "pci_support")] #[cfg(feature = "pci_support")]
@ -2390,11 +2392,11 @@ impl DeviceManager {
&mut self, &mut self,
pci: &mut PciBus, pci: &mut PciBus,
interrupt_manager: &Arc<dyn InterruptManager<GroupConfig = MsiIrqGroupConfig>>, interrupt_manager: &Arc<dyn InterruptManager<GroupConfig = MsiIrqGroupConfig>>,
device_fd: &Arc<DeviceFd>, device: &Arc<dyn hypervisor::Device>,
device_cfg: &mut DeviceConfig, device_cfg: &mut DeviceConfig,
) -> DeviceManagerResult<(u32, String)> { ) -> DeviceManagerResult<(u32, String)> {
#[cfg(feature = "kvm")] #[cfg(feature = "kvm")]
return self.add_vfio_device(pci, interrupt_manager, device_fd, device_cfg); return self.add_vfio_device(pci, interrupt_manager, device, device_cfg);
#[cfg(not(feature = "kvm"))] #[cfg(not(feature = "kvm"))]
Err(DeviceManagerError::NoDevicePassthroughSupport) Err(DeviceManagerError::NoDevicePassthroughSupport)
@ -2405,7 +2407,7 @@ impl DeviceManager {
&mut self, &mut self,
pci: &mut PciBus, pci: &mut PciBus,
interrupt_manager: &Arc<dyn InterruptManager<GroupConfig = MsiIrqGroupConfig>>, interrupt_manager: &Arc<dyn InterruptManager<GroupConfig = MsiIrqGroupConfig>>,
device_fd: &Arc<DeviceFd>, device: &Arc<dyn hypervisor::Device>,
device_cfg: &mut DeviceConfig, device_cfg: &mut DeviceConfig,
) -> DeviceManagerResult<(u32, String)> { ) -> DeviceManagerResult<(u32, String)> {
// We need to shift the device id since the 3 first bits // We need to shift the device id since the 3 first bits
@ -2420,7 +2422,10 @@ impl DeviceManager {
let memory = self.memory_manager.lock().unwrap().guest_memory(); let memory = self.memory_manager.lock().unwrap().guest_memory();
let vfio_container = Arc::new( let vfio_container = Arc::new(
VfioContainer::new(device_fd.clone()).map_err(DeviceManagerError::VfioCreate)?, VfioContainer::new(Arc::new(unsafe {
DeviceFd::from_raw_fd(device.as_raw_fd())
}))
.map_err(DeviceManagerError::VfioCreate)?,
); );
let vfio_device = VfioDevice::new( let vfio_device = VfioDevice::new(
@ -2511,17 +2516,16 @@ impl DeviceManager {
if let Some(device_list_cfg) = &mut devices { if let Some(device_list_cfg) = &mut devices {
// Create the passthrough device handle // Create the passthrough device handle
let device_fd = self let device = self
.address_manager .address_manager
.vm .vm
.create_passthrough_device() .create_passthrough_device()
.map_err(|e| DeviceManagerError::CreatePassthroughDevice(e.into()))?; .map_err(|e| DeviceManagerError::CreatePassthroughDevice(e.into()))?;
let device_fd = Arc::new(device_fd); self.passthrough_device = Some(Arc::clone(&device));
self.passthrough_device = Some(Arc::clone(&device_fd));
for device_cfg in device_list_cfg.iter_mut() { for device_cfg in device_list_cfg.iter_mut() {
let (device_id, _) = let (device_id, _) =
self.add_passthrough_device(pci, interrupt_manager, &device_fd, device_cfg)?; self.add_passthrough_device(pci, interrupt_manager, &device, device_cfg)?;
if device_cfg.iommu && self.iommu_device.is_some() { if device_cfg.iommu && self.iommu_device.is_some() {
iommu_attached_device_ids.push(device_id); iommu_attached_device_ids.push(device_id);
} }
@ -2936,26 +2940,25 @@ impl DeviceManager {
let interrupt_manager = Arc::clone(&self.msi_interrupt_manager); let interrupt_manager = Arc::clone(&self.msi_interrupt_manager);
let device_fd = if let Some(device_fd) = &self.passthrough_device { let device = if let Some(device) = &self.passthrough_device {
Arc::clone(&device_fd) Arc::clone(&device)
} else { } else {
// If the passthrough device file descriptor has not been created yet, // If the passthrough device file descriptor has not been created yet,
// it is created here and stored in the DeviceManager structure for // it is created here and stored in the DeviceManager structure for
// future needs. // future needs.
let device_fd = self let device = self
.address_manager .address_manager
.vm .vm
.create_passthrough_device() .create_passthrough_device()
.map_err(|e| DeviceManagerError::CreatePassthroughDevice(e.into()))?; .map_err(|e| DeviceManagerError::CreatePassthroughDevice(e.into()))?;
let device_fd = Arc::new(device_fd); self.passthrough_device = Some(Arc::clone(&device));
self.passthrough_device = Some(Arc::clone(&device_fd)); device
device_fd
}; };
let (device_id, device_name) = self.add_passthrough_device( let (device_id, device_name) = self.add_passthrough_device(
&mut pci.lock().unwrap(), &mut pci.lock().unwrap(),
&interrupt_manager, &interrupt_manager,
&device_fd, &device,
device_cfg, device_cfg,
)?; )?;