mirror of
https://github.com/cloud-hypervisor/cloud-hypervisor.git
synced 2025-01-22 04:25:21 +00:00
vm-virtio: Implement reset() for vhost-user-fs
The virtio specification defines a device can be reset, which was not supported by this vhost-user-fs implementation. The reason it is needed is to support unbinding this device from the guest driver, and rebind it to vfio-pci driver. Signed-off-by: Sebastien Boeuf <sebastien.boeuf@intel.com>
This commit is contained in:
parent
4b1328a29c
commit
85e1865cb5
@ -1,7 +1,7 @@
|
|||||||
// Copyright 2019 Intel Corporation. All Rights Reserved.
|
// Copyright 2019 Intel Corporation. All Rights Reserved.
|
||||||
// SPDX-License-Identifier: Apache-2.0
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
|
||||||
use super::vu_common_ctrl::setup_vhost_user;
|
use super::vu_common_ctrl::{reset_vhost_user, setup_vhost_user};
|
||||||
use super::{Error, Result};
|
use super::{Error, Result};
|
||||||
use crate::vhost_user::handler::{VhostUserEpollConfig, VhostUserEpollHandler};
|
use crate::vhost_user::handler::{VhostUserEpollConfig, VhostUserEpollHandler};
|
||||||
use crate::{
|
use crate::{
|
||||||
@ -118,6 +118,8 @@ pub struct Fs {
|
|||||||
kill_evt: Option<EventFd>,
|
kill_evt: Option<EventFd>,
|
||||||
cache: Option<(VirtioSharedMemoryList, u64)>,
|
cache: Option<(VirtioSharedMemoryList, u64)>,
|
||||||
slave_req_support: bool,
|
slave_req_support: bool,
|
||||||
|
queue_evts: Option<Vec<EventFd>>,
|
||||||
|
interrupt_cb: Option<Arc<VirtioInterrupt>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Fs {
|
impl Fs {
|
||||||
@ -199,6 +201,8 @@ impl Fs {
|
|||||||
kill_evt: None,
|
kill_evt: None,
|
||||||
cache,
|
cache,
|
||||||
slave_req_support,
|
slave_req_support,
|
||||||
|
queue_evts: None,
|
||||||
|
interrupt_cb: None,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -305,6 +309,21 @@ impl VirtioDevice for Fs {
|
|||||||
};
|
};
|
||||||
self.kill_evt = Some(self_kill_evt);
|
self.kill_evt = Some(self_kill_evt);
|
||||||
|
|
||||||
|
// Save the interrupt EventFD as we need to return it on reset
|
||||||
|
// but clone it to pass into the thread.
|
||||||
|
self.interrupt_cb = Some(interrupt_cb.clone());
|
||||||
|
|
||||||
|
let mut tmp_queue_evts: Vec<EventFd> = Vec::new();
|
||||||
|
for queue_evt in queue_evts.iter() {
|
||||||
|
// Save the queue EventFD as we need to return it on reset
|
||||||
|
// but clone it to pass into the thread.
|
||||||
|
tmp_queue_evts.push(queue_evt.try_clone().map_err(|e| {
|
||||||
|
error!("failed to clone queue EventFd: {}", e);
|
||||||
|
ActivateError::BadActivate
|
||||||
|
})?);
|
||||||
|
}
|
||||||
|
self.queue_evts = Some(tmp_queue_evts);
|
||||||
|
|
||||||
let vu_call_evt_queue_list = setup_vhost_user(
|
let vu_call_evt_queue_list = setup_vhost_user(
|
||||||
&mut self.vu,
|
&mut self.vu,
|
||||||
&mem.read().unwrap(),
|
&mem.read().unwrap(),
|
||||||
@ -361,6 +380,24 @@ impl VirtioDevice for Fs {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn reset(&mut self) -> Option<(Arc<VirtioInterrupt>, Vec<EventFd>)> {
|
||||||
|
if let Err(e) = reset_vhost_user(&mut self.vu, self.queue_sizes.len()) {
|
||||||
|
error!("Failed to reset vhost-user daemon: {:?}", e);
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(kill_evt) = self.kill_evt.take() {
|
||||||
|
// Ignore the result because there is nothing we can do about it.
|
||||||
|
let _ = kill_evt.write(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return the interrupt and queue EventFDs
|
||||||
|
Some((
|
||||||
|
self.interrupt_cb.take().unwrap(),
|
||||||
|
self.queue_evts.take().unwrap(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
fn get_shm_regions(&self) -> Option<VirtioSharedMemoryList> {
|
fn get_shm_regions(&self) -> Option<VirtioSharedMemoryList> {
|
||||||
if let Some(cache) = self.cache.clone() {
|
if let Some(cache) = self.cache.clone() {
|
||||||
Some(cache.0)
|
Some(cache.0)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user