stratovirt/0019-vfio-fix-hot-plug-the-same-device-multiple-times-pro.patch
Jie Yang c98bb6b512 Update version to 2.1.0-4
Signed-off-by: Jie Yang <yangjieyj.yang@huawei.com>
(cherry picked from commit e9cfa81784d4a86efd4392a487c0294d440c03c0)
2022-03-13 12:08:05 +08:00

74 lines
2.7 KiB
Diff

From 13e120e99398ece2ceab00f90d0087123309ea60 Mon Sep 17 00:00:00 2001
From: "Xinle.Guo" <guoxinle1@huawei.com>
Date: Fri, 11 Mar 2022 14:50:48 +0800
Subject: [PATCH 4/8] vfio: fix hot plug the same device multiple times problem
It will success to hot plug the same device after the second
time. But this is not what we expected. So, we check if the
device name is repeated before get VFIO device.
Signed-off-by: Xinle.Guo <guoxinle1@huawei.com>
---
vfio/src/vfio_dev.rs | 17 ++++++++++++++---
1 file changed, 14 insertions(+), 3 deletions(-)
diff --git a/vfio/src/vfio_dev.rs b/vfio/src/vfio_dev.rs
index 9607b71..198421c 100644
--- a/vfio/src/vfio_dev.rs
+++ b/vfio/src/vfio_dev.rs
@@ -464,6 +464,8 @@ pub struct VfioDevInfo {
pub struct VfioDevice {
/// File descriptor for a VFIO device instance.
pub fd: File,
+ /// Identify the unique VFIO device.
+ pub name: String,
/// Vfio group the device belongs to.
pub group: Weak<VfioGroup>,
/// Vfio container the device belongs to.
@@ -515,10 +517,12 @@ impl VfioDevice {
let group =
Self::vfio_get_group(&path, mem_as).chain_err(|| "Failed to get iommu group")?;
- let fd = Self::vfio_get_device(&group, &path).chain_err(|| "Failed to get vfio device")?;
+ let (name, fd) =
+ Self::vfio_get_device(&group, &path).chain_err(|| "Failed to get vfio device")?;
let dev_info = Self::get_dev_info(&fd).chain_err(|| "Failed to get device info")?;
let vfio_dev = Arc::new(Mutex::new(VfioDevice {
fd,
+ name,
group: Arc::downgrade(&group),
container: group.container.clone(),
dev_info,
@@ -569,11 +573,18 @@ impl VfioDevice {
Ok(group)
}
- fn vfio_get_device(group: &VfioGroup, name: &Path) -> Result<File> {
+ fn vfio_get_device(group: &VfioGroup, name: &Path) -> Result<(String, File)> {
let mut dev_name: &str = "";
if let Some(n) = name.file_name() {
dev_name = n.to_str().chain_err(|| "Invalid device path")?;
}
+
+ for device in group.devices.lock().unwrap().iter() {
+ if device.1.lock().unwrap().name == dev_name {
+ bail!("Device {} is already attached", dev_name);
+ }
+ }
+
let path: CString = CString::new(dev_name.as_bytes())
.chain_err(|| "Failed to convert device name to CString type of data")?;
let ptr = path.as_ptr();
@@ -589,7 +600,7 @@ impl VfioDevice {
// Safe as we have verified that fd is a valid FD.
let device = unsafe { File::from_raw_fd(fd) };
- Ok(device)
+ Ok((String::from(dev_name), device))
}
fn get_dev_info(device: &File) -> Result<VfioDevInfo> {
--
2.20.1