mirror of
https://github.com/cloud-hypervisor/cloud-hypervisor.git
synced 2025-01-11 07:07:42 +00:00
vmm: Add ability to add virtio-fs device post-boot
Adds DeviceManager method `make_virtio_fs_device` which creates a single device, and modifies `make_virtio_fs_devices` to use this method. Implements the new `vm.add-fs route`. Signed-off-by: Dean Sheather <dean@coder.com>
This commit is contained in:
parent
bb2139a408
commit
c2abadc293
@ -5,11 +5,11 @@
|
|||||||
|
|
||||||
use crate::api::http::EndpointHandler;
|
use crate::api::http::EndpointHandler;
|
||||||
use crate::api::{
|
use crate::api::{
|
||||||
vm_add_device, vm_add_disk, vm_add_net, vm_add_pmem, vm_boot, vm_create, vm_delete, vm_info,
|
vm_add_device, vm_add_disk, vm_add_fs, vm_add_net, vm_add_pmem, vm_boot, vm_create, vm_delete,
|
||||||
vm_pause, vm_reboot, vm_remove_device, vm_resize, vm_restore, vm_resume, vm_shutdown,
|
vm_info, vm_pause, vm_reboot, vm_remove_device, vm_resize, vm_restore, vm_resume, vm_shutdown,
|
||||||
vm_snapshot, vmm_ping, vmm_shutdown, ApiError, ApiRequest, ApiResult, DeviceConfig, DiskConfig,
|
vm_snapshot, vmm_ping, vmm_shutdown, ApiError, ApiRequest, ApiResult, DeviceConfig, DiskConfig,
|
||||||
NetConfig, PmemConfig, RestoreConfig, VmAction, VmConfig, VmRemoveDeviceData, VmResizeData,
|
FsConfig, NetConfig, PmemConfig, RestoreConfig, VmAction, VmConfig, VmRemoveDeviceData,
|
||||||
VmSnapshotConfig,
|
VmResizeData, VmSnapshotConfig,
|
||||||
};
|
};
|
||||||
use micro_http::{Body, Method, Request, Response, StatusCode, Version};
|
use micro_http::{Body, Method, Request, Response, StatusCode, Version};
|
||||||
use serde_json::Error as SerdeError;
|
use serde_json::Error as SerdeError;
|
||||||
@ -71,6 +71,9 @@ pub enum HttpError {
|
|||||||
/// Could not add a disk to a VM
|
/// Could not add a disk to a VM
|
||||||
VmAddDisk(ApiError),
|
VmAddDisk(ApiError),
|
||||||
|
|
||||||
|
/// Could not add a fs to a VM
|
||||||
|
VmAddFs(ApiError),
|
||||||
|
|
||||||
/// Could not add a pmem device to a VM
|
/// Could not add a pmem device to a VM
|
||||||
VmAddPmem(ApiError),
|
VmAddPmem(ApiError),
|
||||||
|
|
||||||
@ -494,12 +497,33 @@ impl EndpointHandler for VmAddFs {
|
|||||||
fn handle_request(
|
fn handle_request(
|
||||||
&self,
|
&self,
|
||||||
req: &Request,
|
req: &Request,
|
||||||
_api_notifier: EventFd,
|
api_notifier: EventFd,
|
||||||
_api_sender: Sender<ApiRequest>,
|
api_sender: Sender<ApiRequest>,
|
||||||
) -> Response {
|
) -> Response {
|
||||||
match req.method() {
|
match req.method() {
|
||||||
// Not implemented.
|
Method::Put => {
|
||||||
Method::Put => Response::new(Version::Http11, StatusCode::NotImplemented),
|
match &req.body {
|
||||||
|
Some(body) => {
|
||||||
|
// Deserialize into a FsConfig
|
||||||
|
let vm_add_fs_data: FsConfig = match serde_json::from_slice(body.raw())
|
||||||
|
.map_err(HttpError::SerdeJsonDeserialize)
|
||||||
|
{
|
||||||
|
Ok(config) => config,
|
||||||
|
Err(e) => return error_response(e, StatusCode::BadRequest),
|
||||||
|
};
|
||||||
|
|
||||||
|
// Call vm_add_fs()
|
||||||
|
match vm_add_fs(api_notifier, api_sender, Arc::new(vm_add_fs_data))
|
||||||
|
.map_err(HttpError::VmAddFs)
|
||||||
|
{
|
||||||
|
Ok(_) => Response::new(Version::Http11, StatusCode::NoContent),
|
||||||
|
Err(e) => error_response(e, StatusCode::InternalServerError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
None => Response::new(Version::Http11, StatusCode::BadRequest),
|
||||||
|
}
|
||||||
|
}
|
||||||
_ => Response::new(Version::Http11, StatusCode::BadRequest),
|
_ => Response::new(Version::Http11, StatusCode::BadRequest),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -37,7 +37,9 @@ pub use self::http::start_http_thread;
|
|||||||
pub mod http;
|
pub mod http;
|
||||||
pub mod http_endpoint;
|
pub mod http_endpoint;
|
||||||
|
|
||||||
use crate::config::{DeviceConfig, DiskConfig, NetConfig, PmemConfig, RestoreConfig, VmConfig};
|
use crate::config::{
|
||||||
|
DeviceConfig, DiskConfig, FsConfig, NetConfig, PmemConfig, RestoreConfig, VmConfig,
|
||||||
|
};
|
||||||
use crate::vm::{Error as VmError, VmState};
|
use crate::vm::{Error as VmError, VmState};
|
||||||
use std::io;
|
use std::io;
|
||||||
use std::sync::mpsc::{channel, RecvError, SendError, Sender};
|
use std::sync::mpsc::{channel, RecvError, SendError, Sender};
|
||||||
@ -122,6 +124,9 @@ pub enum ApiError {
|
|||||||
/// The disk could not be added to the VM.
|
/// The disk could not be added to the VM.
|
||||||
VmAddDisk(VmError),
|
VmAddDisk(VmError),
|
||||||
|
|
||||||
|
/// The fs could not be added to the VM.
|
||||||
|
VmAddFs(VmError),
|
||||||
|
|
||||||
/// The pmem device could not be added to the VM.
|
/// The pmem device could not be added to the VM.
|
||||||
VmAddPmem(VmError),
|
VmAddPmem(VmError),
|
||||||
|
|
||||||
@ -230,6 +235,9 @@ pub enum ApiRequest {
|
|||||||
/// Add a disk to the VM.
|
/// Add a disk to the VM.
|
||||||
VmAddDisk(Arc<DiskConfig>, Sender<ApiResponse>),
|
VmAddDisk(Arc<DiskConfig>, Sender<ApiResponse>),
|
||||||
|
|
||||||
|
/// Add a fs to the VM.
|
||||||
|
VmAddFs(Arc<FsConfig>, Sender<ApiResponse>),
|
||||||
|
|
||||||
/// Add a pmem device to the VM.
|
/// Add a pmem device to the VM.
|
||||||
VmAddPmem(Arc<PmemConfig>, Sender<ApiResponse>),
|
VmAddPmem(Arc<PmemConfig>, Sender<ApiResponse>),
|
||||||
|
|
||||||
@ -484,6 +492,24 @@ pub fn vm_add_disk(
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn vm_add_fs(
|
||||||
|
api_evt: EventFd,
|
||||||
|
api_sender: Sender<ApiRequest>,
|
||||||
|
data: Arc<FsConfig>,
|
||||||
|
) -> ApiResult<()> {
|
||||||
|
let (response_sender, response_receiver) = channel();
|
||||||
|
|
||||||
|
// Send the VM add-fs request.
|
||||||
|
api_sender
|
||||||
|
.send(ApiRequest::VmAddFs(data, response_sender))
|
||||||
|
.map_err(ApiError::RequestSend)?;
|
||||||
|
api_evt.write(1).map_err(ApiError::EventFdWrite)?;
|
||||||
|
|
||||||
|
response_receiver.recv().map_err(ApiError::ResponseRecv)??;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
pub fn vm_add_pmem(
|
pub fn vm_add_pmem(
|
||||||
api_evt: EventFd,
|
api_evt: EventFd,
|
||||||
api_sender: Sender<ApiRequest>,
|
api_sender: Sender<ApiRequest>,
|
||||||
|
@ -14,7 +14,7 @@ extern crate vm_device;
|
|||||||
use crate::config::ConsoleOutputMode;
|
use crate::config::ConsoleOutputMode;
|
||||||
#[cfg(feature = "pci_support")]
|
#[cfg(feature = "pci_support")]
|
||||||
use crate::config::DeviceConfig;
|
use crate::config::DeviceConfig;
|
||||||
use crate::config::{DiskConfig, NetConfig, PmemConfig, VmConfig};
|
use crate::config::{DiskConfig, FsConfig, NetConfig, PmemConfig, VmConfig};
|
||||||
use crate::interrupt::{
|
use crate::interrupt::{
|
||||||
KvmLegacyUserspaceInterruptManager, KvmMsiInterruptManager, KvmRoutingEntry,
|
KvmLegacyUserspaceInterruptManager, KvmMsiInterruptManager, KvmRoutingEntry,
|
||||||
};
|
};
|
||||||
@ -104,6 +104,9 @@ pub enum DeviceManagerError {
|
|||||||
/// Cannot create virtio-fs device
|
/// Cannot create virtio-fs device
|
||||||
CreateVirtioFs(vm_virtio::vhost_user::Error),
|
CreateVirtioFs(vm_virtio::vhost_user::Error),
|
||||||
|
|
||||||
|
/// Virtio-fs device was created without a sock.
|
||||||
|
NoVirtioFsSock,
|
||||||
|
|
||||||
/// Cannot create vhost-user-blk device
|
/// Cannot create vhost-user-blk device
|
||||||
CreateVhostUserBlk(vm_virtio::vhost_user::Error),
|
CreateVhostUserBlk(vm_virtio::vhost_user::Error),
|
||||||
|
|
||||||
@ -1414,13 +1417,10 @@ impl DeviceManager {
|
|||||||
Ok(devices)
|
Ok(devices)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn make_virtio_fs_devices(
|
fn make_virtio_fs_device(
|
||||||
&mut self,
|
&mut self,
|
||||||
) -> DeviceManagerResult<Vec<(VirtioDeviceArc, bool, Option<String>)>> {
|
fs_cfg: &FsConfig,
|
||||||
let mut devices = Vec::new();
|
) -> DeviceManagerResult<(VirtioDeviceArc, bool, Option<String>)> {
|
||||||
// Add virtio-fs if required
|
|
||||||
if let Some(fs_list_cfg) = &self.config.lock().unwrap().fs {
|
|
||||||
for fs_cfg in fs_list_cfg.iter() {
|
|
||||||
if let Some(fs_sock) = fs_cfg.sock.to_str() {
|
if let Some(fs_sock) = fs_cfg.sock.to_str() {
|
||||||
let cache = if fs_cfg.dax {
|
let cache = if fs_cfg.dax {
|
||||||
let fs_cache = fs_cfg.cache_size;
|
let fs_cache = fs_cfg.cache_size;
|
||||||
@ -1431,11 +1431,7 @@ impl DeviceManager {
|
|||||||
.allocator
|
.allocator
|
||||||
.lock()
|
.lock()
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.allocate_mmio_addresses(
|
.allocate_mmio_addresses(None, fs_cache as GuestUsize, Some(0x0020_0000))
|
||||||
None,
|
|
||||||
fs_cache as GuestUsize,
|
|
||||||
Some(0x0020_0000),
|
|
||||||
)
|
|
||||||
.ok_or(DeviceManagerError::FsRangeAllocation)?;
|
.ok_or(DeviceManagerError::FsRangeAllocation)?;
|
||||||
|
|
||||||
let mmap_region = MmapRegion::build(
|
let mmap_region = MmapRegion::build(
|
||||||
@ -1491,17 +1487,28 @@ impl DeviceManager {
|
|||||||
.map_err(DeviceManagerError::CreateVirtioFs)?,
|
.map_err(DeviceManagerError::CreateVirtioFs)?,
|
||||||
));
|
));
|
||||||
|
|
||||||
devices.push((
|
self.add_migratable_device(Arc::clone(&virtio_fs_device) as Arc<Mutex<dyn Migratable>>);
|
||||||
|
|
||||||
|
Ok((
|
||||||
Arc::clone(&virtio_fs_device) as VirtioDeviceArc,
|
Arc::clone(&virtio_fs_device) as VirtioDeviceArc,
|
||||||
false,
|
false,
|
||||||
None,
|
None,
|
||||||
));
|
))
|
||||||
|
} else {
|
||||||
let migratable = Arc::clone(&virtio_fs_device) as Arc<Mutex<dyn Migratable>>;
|
Err(DeviceManagerError::NoVirtioFsSock)
|
||||||
let id = migratable.lock().unwrap().id();
|
|
||||||
self.migratable_devices.push((id, migratable));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn make_virtio_fs_devices(
|
||||||
|
&mut self,
|
||||||
|
) -> DeviceManagerResult<Vec<(VirtioDeviceArc, bool, Option<String>)>> {
|
||||||
|
let mut devices = Vec::new();
|
||||||
|
|
||||||
|
let fs_devices = self.config.lock().unwrap().fs.clone();
|
||||||
|
if let Some(fs_list_cfg) = &fs_devices {
|
||||||
|
for fs_cfg in fs_list_cfg.iter() {
|
||||||
|
devices.push(self.make_virtio_fs_device(fs_cfg)?);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(devices)
|
Ok(devices)
|
||||||
@ -2248,6 +2255,12 @@ impl DeviceManager {
|
|||||||
self.hotplug_virtio_pci_device(device, iommu_attached, id)
|
self.hotplug_virtio_pci_device(device, iommu_attached, id)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "pci_support")]
|
||||||
|
pub fn add_fs(&mut self, fs_cfg: &mut FsConfig) -> DeviceManagerResult<()> {
|
||||||
|
let (device, iommu_attached, id) = self.make_virtio_fs_device(fs_cfg)?;
|
||||||
|
self.hotplug_virtio_pci_device(device, iommu_attached, id)
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(feature = "pci_support")]
|
#[cfg(feature = "pci_support")]
|
||||||
pub fn add_pmem(&mut self, pmem_cfg: &mut PmemConfig) -> DeviceManagerResult<()> {
|
pub fn add_pmem(&mut self, pmem_cfg: &mut PmemConfig) -> DeviceManagerResult<()> {
|
||||||
let (device, iommu_attached, id) = self.make_virtio_pmem_device(pmem_cfg)?;
|
let (device, iommu_attached, id) = self.make_virtio_pmem_device(pmem_cfg)?;
|
||||||
|
@ -18,7 +18,9 @@ extern crate url;
|
|||||||
extern crate vmm_sys_util;
|
extern crate vmm_sys_util;
|
||||||
|
|
||||||
use crate::api::{ApiError, ApiRequest, ApiResponse, ApiResponsePayload, VmInfo, VmmPingResponse};
|
use crate::api::{ApiError, ApiRequest, ApiResponse, ApiResponsePayload, VmInfo, VmmPingResponse};
|
||||||
use crate::config::{DeviceConfig, DiskConfig, NetConfig, PmemConfig, RestoreConfig, VmConfig};
|
use crate::config::{
|
||||||
|
DeviceConfig, DiskConfig, FsConfig, NetConfig, PmemConfig, RestoreConfig, VmConfig,
|
||||||
|
};
|
||||||
use crate::migration::{recv_vm_snapshot, vm_config_from_snapshot};
|
use crate::migration::{recv_vm_snapshot, vm_config_from_snapshot};
|
||||||
use crate::seccomp_filters::{get_seccomp_filter, Thread};
|
use crate::seccomp_filters::{get_seccomp_filter, Thread};
|
||||||
use crate::vm::{Error as VmError, Vm, VmState};
|
use crate::vm::{Error as VmError, Vm, VmState};
|
||||||
@ -483,6 +485,19 @@ impl Vmm {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn vm_add_fs(&mut self, fs_cfg: FsConfig) -> result::Result<(), VmError> {
|
||||||
|
if let Some(ref mut vm) = self.vm {
|
||||||
|
if let Err(e) = vm.add_fs(fs_cfg) {
|
||||||
|
error!("Error when adding new fs to the VM: {:?}", e);
|
||||||
|
Err(e)
|
||||||
|
} else {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Err(VmError::VmNotRunning)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn vm_add_pmem(&mut self, pmem_cfg: PmemConfig) -> result::Result<(), VmError> {
|
fn vm_add_pmem(&mut self, pmem_cfg: PmemConfig) -> result::Result<(), VmError> {
|
||||||
if let Some(ref mut vm) = self.vm {
|
if let Some(ref mut vm) = self.vm {
|
||||||
if let Err(e) = vm.add_pmem(pmem_cfg) {
|
if let Err(e) = vm.add_pmem(pmem_cfg) {
|
||||||
@ -701,6 +716,13 @@ impl Vmm {
|
|||||||
.map(|_| ApiResponsePayload::Empty);
|
.map(|_| ApiResponsePayload::Empty);
|
||||||
sender.send(response).map_err(Error::ApiResponseSend)?;
|
sender.send(response).map_err(Error::ApiResponseSend)?;
|
||||||
}
|
}
|
||||||
|
ApiRequest::VmAddFs(add_fs_data, sender) => {
|
||||||
|
let response = self
|
||||||
|
.vm_add_fs(add_fs_data.as_ref().clone())
|
||||||
|
.map_err(ApiError::VmAddFs)
|
||||||
|
.map(|_| ApiResponsePayload::Empty);
|
||||||
|
sender.send(response).map_err(Error::ApiResponseSend)?;
|
||||||
|
}
|
||||||
ApiRequest::VmAddPmem(add_pmem_data, sender) => {
|
ApiRequest::VmAddPmem(add_pmem_data, sender) => {
|
||||||
let response = self
|
let response = self
|
||||||
.vm_add_pmem(add_pmem_data.as_ref().clone())
|
.vm_add_pmem(add_pmem_data.as_ref().clone())
|
||||||
|
@ -26,7 +26,8 @@ extern crate vm_memory;
|
|||||||
extern crate vm_virtio;
|
extern crate vm_virtio;
|
||||||
|
|
||||||
use crate::config::{
|
use crate::config::{
|
||||||
DeviceConfig, DiskConfig, HotplugMethod, NetConfig, PmemConfig, ValidationError, VmConfig,
|
DeviceConfig, DiskConfig, FsConfig, HotplugMethod, NetConfig, PmemConfig, ValidationError,
|
||||||
|
VmConfig,
|
||||||
};
|
};
|
||||||
use crate::cpu;
|
use crate::cpu;
|
||||||
use crate::device_manager::{get_win_size, Console, DeviceManager, DeviceManagerError};
|
use crate::device_manager::{get_win_size, Console, DeviceManager, DeviceManagerError};
|
||||||
@ -777,6 +778,39 @@ impl Vm {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn add_fs(&mut self, mut _fs_cfg: FsConfig) -> Result<()> {
|
||||||
|
if cfg!(feature = "pci_support") {
|
||||||
|
#[cfg(feature = "pci_support")]
|
||||||
|
{
|
||||||
|
self.device_manager
|
||||||
|
.lock()
|
||||||
|
.unwrap()
|
||||||
|
.add_fs(&mut _fs_cfg)
|
||||||
|
.map_err(Error::DeviceManager)?;
|
||||||
|
|
||||||
|
// Update VmConfig by adding the new device. This is important to
|
||||||
|
// ensure the device would be created in case of a reboot.
|
||||||
|
{
|
||||||
|
let mut config = self.config.lock().unwrap();
|
||||||
|
if let Some(fs_config) = config.fs.as_mut() {
|
||||||
|
fs_config.push(_fs_cfg);
|
||||||
|
} else {
|
||||||
|
config.fs = Some(vec![_fs_cfg]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
self.device_manager
|
||||||
|
.lock()
|
||||||
|
.unwrap()
|
||||||
|
.notify_hotplug(HotPlugNotificationFlags::PCI_DEVICES_CHANGED)
|
||||||
|
.map_err(Error::DeviceManager)?;
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(Error::NoPciSupport)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn add_pmem(&mut self, mut _pmem_cfg: PmemConfig) -> Result<()> {
|
pub fn add_pmem(&mut self, mut _pmem_cfg: PmemConfig) -> Result<()> {
|
||||||
if cfg!(feature = "pci_support") {
|
if cfg!(feature = "pci_support") {
|
||||||
#[cfg(feature = "pci_support")]
|
#[cfg(feature = "pci_support")]
|
||||||
|
Loading…
Reference in New Issue
Block a user