mirror of
https://github.com/cloud-hypervisor/cloud-hypervisor.git
synced 2024-12-22 05:35:20 +00:00
vm-migration: Snapshot should have a unique SnapshotDataSection
There's no reason to carry a HashMap of SnapshotDataSection per Snapshot. And given we now provide at most one SnapshotDataSection per Snapshot, there's no need to keep the id part of the SnapshotDataSection structure. Signed-off-by: Sebastien Boeuf <sebastien.boeuf@intel.com>
This commit is contained in:
parent
0489b6314e
commit
5b3bcfa233
@ -465,7 +465,7 @@ impl VfioCommon {
|
||||
|
||||
let state: Option<VfioCommonState> = snapshot
|
||||
.as_ref()
|
||||
.map(|s| s.to_versioned_state(VFIO_COMMON_ID))
|
||||
.map(|s| s.to_versioned_state())
|
||||
.transpose()
|
||||
.map_err(|e| {
|
||||
VfioPciError::RetrieveVfioCommonState(anyhow!(
|
||||
|
@ -522,7 +522,7 @@ impl VirtioPciDevice {
|
||||
|
||||
let state: Option<VirtioPciDeviceState> = snapshot
|
||||
.as_ref()
|
||||
.map(|s| s.to_versioned_state(&id))
|
||||
.map(|s| s.to_versioned_state())
|
||||
.transpose()
|
||||
.map_err(|e| {
|
||||
VirtioPciDeviceError::CreateVirtioPciDevice(anyhow!(
|
||||
|
@ -81,9 +81,6 @@ pub trait Pausable {
|
||||
/// allows for easier and forward compatible extensions.
|
||||
#[derive(Clone, Default, Deserialize, Serialize)]
|
||||
pub struct SnapshotDataSection {
|
||||
/// The section id.
|
||||
pub id: String,
|
||||
|
||||
/// The section serialized snapshot.
|
||||
pub snapshot: Vec<u8>,
|
||||
}
|
||||
@ -94,9 +91,8 @@ impl SnapshotDataSection {
|
||||
where
|
||||
T: Deserialize<'a>,
|
||||
{
|
||||
serde_json::from_slice(&self.snapshot).map_err(|e| {
|
||||
MigratableError::Restore(anyhow!("Error deserialising: {} {}", self.id, e))
|
||||
})
|
||||
serde_json::from_slice(&self.snapshot)
|
||||
.map_err(|e| MigratableError::Restore(anyhow!("Error deserialising: {}", e)))
|
||||
}
|
||||
|
||||
/// Generate versioned state
|
||||
@ -109,39 +105,33 @@ impl SnapshotDataSection {
|
||||
&T::version_map(),
|
||||
VMM_VERSION,
|
||||
)
|
||||
.map_err(|e| MigratableError::Restore(anyhow!("Error deserialising: {} {}", self.id, e)))
|
||||
.map_err(|e| MigratableError::Restore(anyhow!("Error deserialising: {}", e)))
|
||||
}
|
||||
|
||||
/// Create from state that can be serialized
|
||||
pub fn new_from_state<T>(id: &str, state: &T) -> Result<Self, MigratableError>
|
||||
pub fn new_from_state<T>(state: &T) -> Result<Self, MigratableError>
|
||||
where
|
||||
T: Serialize,
|
||||
{
|
||||
let snapshot = serde_json::to_vec(state)
|
||||
.map_err(|e| MigratableError::Snapshot(anyhow!("Error serialising: {} {}", id, e)))?;
|
||||
.map_err(|e| MigratableError::Snapshot(anyhow!("Error serialising: {}", e)))?;
|
||||
|
||||
let snapshot_data = SnapshotDataSection {
|
||||
id: format!("{}-section", id),
|
||||
snapshot,
|
||||
};
|
||||
let snapshot_data = SnapshotDataSection { snapshot };
|
||||
|
||||
Ok(snapshot_data)
|
||||
}
|
||||
|
||||
/// Create from versioned state
|
||||
pub fn new_from_versioned_state<T>(id: &str, state: &T) -> Result<Self, MigratableError>
|
||||
pub fn new_from_versioned_state<T>(state: &T) -> Result<Self, MigratableError>
|
||||
where
|
||||
T: Versionize + VersionMapped,
|
||||
{
|
||||
let mut snapshot = Vec::new();
|
||||
state
|
||||
.serialize(&mut snapshot, &T::version_map(), VMM_VERSION)
|
||||
.map_err(|e| MigratableError::Snapshot(anyhow!("Error serialising: {} {}", id, e)))?;
|
||||
.map_err(|e| MigratableError::Snapshot(anyhow!("Error serialising: {}", e)))?;
|
||||
|
||||
let snapshot_data = SnapshotDataSection {
|
||||
id: format!("{}-section", id),
|
||||
snapshot,
|
||||
};
|
||||
let snapshot_data = SnapshotDataSection { snapshot };
|
||||
|
||||
Ok(snapshot_data)
|
||||
}
|
||||
@ -168,7 +158,7 @@ pub struct Snapshot {
|
||||
|
||||
/// The Snapshottable component's snapshot data.
|
||||
/// A map of snapshot sections, indexed by the section ids.
|
||||
pub snapshot_data: std::collections::HashMap<String, SnapshotDataSection>,
|
||||
pub snapshot_data: Option<SnapshotDataSection>,
|
||||
}
|
||||
|
||||
impl Snapshot {
|
||||
@ -186,7 +176,7 @@ impl Snapshot {
|
||||
T: Serialize,
|
||||
{
|
||||
let mut snapshot_data = Snapshot::new(id);
|
||||
snapshot_data.add_data_section(SnapshotDataSection::new_from_state(id, state)?);
|
||||
snapshot_data.add_data_section(SnapshotDataSection::new_from_state(state)?);
|
||||
|
||||
Ok(snapshot_data)
|
||||
}
|
||||
@ -197,7 +187,7 @@ impl Snapshot {
|
||||
T: Versionize + VersionMapped,
|
||||
{
|
||||
let mut snapshot_data = Snapshot::new(id);
|
||||
snapshot_data.add_data_section(SnapshotDataSection::new_from_versioned_state(id, state)?);
|
||||
snapshot_data.add_data_section(SnapshotDataSection::new_from_versioned_state(state)?);
|
||||
|
||||
Ok(snapshot_data)
|
||||
}
|
||||
@ -210,28 +200,28 @@ impl Snapshot {
|
||||
|
||||
/// Add a SnapshotDatasection to the component snapshot data.
|
||||
pub fn add_data_section(&mut self, section: SnapshotDataSection) {
|
||||
self.snapshot_data.insert(section.id.clone(), section);
|
||||
self.snapshot_data = Some(section);
|
||||
}
|
||||
|
||||
/// Generate the state data from the snapshot
|
||||
pub fn to_state<'a, T>(&'a self, id: &str) -> Result<T, MigratableError>
|
||||
pub fn to_state<'a, T>(&'a self) -> Result<T, MigratableError>
|
||||
where
|
||||
T: Deserialize<'a>,
|
||||
{
|
||||
self.snapshot_data
|
||||
.get(&format!("{}-section", id))
|
||||
.ok_or_else(|| MigratableError::Restore(anyhow!("Missing section for {}", id)))?
|
||||
.as_ref()
|
||||
.ok_or_else(|| MigratableError::Restore(anyhow!("Missing snapshot data")))?
|
||||
.to_state()
|
||||
}
|
||||
|
||||
/// Generate versioned state
|
||||
pub fn to_versioned_state<T>(&self, id: &str) -> Result<T, MigratableError>
|
||||
pub fn to_versioned_state<T>(&self) -> Result<T, MigratableError>
|
||||
where
|
||||
T: Versionize + VersionMapped,
|
||||
{
|
||||
self.snapshot_data
|
||||
.get(&format!("{}-section", id))
|
||||
.ok_or_else(|| MigratableError::Restore(anyhow!("Missing section for {}", id)))?
|
||||
.as_ref()
|
||||
.ok_or_else(|| MigratableError::Restore(anyhow!("Missing snapshot data")))?
|
||||
.to_versioned_state()
|
||||
}
|
||||
}
|
||||
@ -249,7 +239,7 @@ where
|
||||
{
|
||||
snapshot
|
||||
.and_then(|s| s.snapshots.get(id).map(|s| *s.clone()))
|
||||
.map(|s| s.to_versioned_state(id))
|
||||
.map(|s| s.to_versioned_state())
|
||||
.transpose()
|
||||
}
|
||||
|
||||
|
@ -403,10 +403,7 @@ impl Snapshottable for Vcpu {
|
||||
// TODO: The special format of the CPU id can be removed once ready to
|
||||
// break live upgrade.
|
||||
let mut vcpu_snapshot = Snapshot::new(&format!("{:03}", self.id));
|
||||
vcpu_snapshot.add_data_section(SnapshotDataSection::new_from_state(
|
||||
VCPU_SNAPSHOT_ID,
|
||||
&saved_state,
|
||||
)?);
|
||||
vcpu_snapshot.add_data_section(SnapshotDataSection::new_from_state(&saved_state)?);
|
||||
|
||||
self.saved_state = Some(saved_state);
|
||||
|
||||
@ -717,7 +714,7 @@ impl CpuManager {
|
||||
#[cfg(target_arch = "aarch64")]
|
||||
vcpu.init(&self.vm)?;
|
||||
|
||||
let state: CpuState = snapshot.to_state(VCPU_SNAPSHOT_ID).map_err(|e| {
|
||||
let state: CpuState = snapshot.to_state().map_err(|e| {
|
||||
Error::VcpuCreate(anyhow!("Could not get vCPU state from snapshot {:?}", e))
|
||||
})?;
|
||||
vcpu.vcpu
|
||||
|
@ -966,7 +966,7 @@ impl DeviceManager {
|
||||
trace_scoped!("DeviceManager::new");
|
||||
|
||||
let (device_tree, device_id_cnt) = if let Some(snapshot) = snapshot.as_ref() {
|
||||
let state: DeviceManagerState = snapshot.to_state(DEVICE_MANAGER_SNAPSHOT_ID).unwrap();
|
||||
let state: DeviceManagerState = snapshot.to_state().unwrap();
|
||||
(
|
||||
Arc::new(Mutex::new(state.device_tree.clone())),
|
||||
state.device_id_cnt,
|
||||
@ -1389,7 +1389,7 @@ impl DeviceManager {
|
||||
}
|
||||
|
||||
let vgic_state = vgic_snapshot
|
||||
.to_state(&id)
|
||||
.to_state()
|
||||
.map_err(DeviceManagerError::RestoreGetState)?;
|
||||
let saved_vcpu_states = self.cpu_manager.lock().unwrap().get_saved_states();
|
||||
interrupt_controller
|
||||
@ -2150,7 +2150,7 @@ impl DeviceManager {
|
||||
.map_err(DeviceManagerError::EventFd)?,
|
||||
self.force_iommu,
|
||||
snapshot
|
||||
.map(|s| s.to_versioned_state(&id))
|
||||
.map(|s| s.to_versioned_state())
|
||||
.transpose()
|
||||
.map_err(DeviceManagerError::RestoreGetState)?,
|
||||
) {
|
||||
@ -2249,7 +2249,7 @@ impl DeviceManager {
|
||||
.try_clone()
|
||||
.map_err(DeviceManagerError::EventFd)?,
|
||||
snapshot
|
||||
.map(|s| s.to_versioned_state(&id))
|
||||
.map(|s| s.to_versioned_state())
|
||||
.transpose()
|
||||
.map_err(DeviceManagerError::RestoreGetState)?,
|
||||
)
|
||||
@ -2332,7 +2332,7 @@ impl DeviceManager {
|
||||
.map_err(DeviceManagerError::EventFd)?,
|
||||
self.force_iommu,
|
||||
snapshot
|
||||
.map(|s| s.to_versioned_state(&id))
|
||||
.map(|s| s.to_versioned_state())
|
||||
.transpose()
|
||||
.map_err(DeviceManagerError::RestoreGetState)?,
|
||||
) {
|
||||
@ -2349,7 +2349,7 @@ impl DeviceManager {
|
||||
)
|
||||
} else {
|
||||
let state = snapshot
|
||||
.map(|s| s.to_versioned_state(&id))
|
||||
.map(|s| s.to_versioned_state())
|
||||
.transpose()
|
||||
.map_err(DeviceManagerError::RestoreGetState)?;
|
||||
|
||||
@ -4456,10 +4456,7 @@ impl Snapshottable for DeviceManager {
|
||||
}
|
||||
|
||||
// Then we store the DeviceManager state.
|
||||
snapshot.add_data_section(SnapshotDataSection::new_from_state(
|
||||
DEVICE_MANAGER_SNAPSHOT_ID,
|
||||
&self.state(),
|
||||
)?);
|
||||
snapshot.add_data_section(SnapshotDataSection::new_from_state(&self.state())?);
|
||||
|
||||
Ok(snapshot)
|
||||
}
|
||||
|
@ -1160,9 +1160,8 @@ impl MemoryManager {
|
||||
let mut memory_file_path = url_to_path(source_url).map_err(Error::Restore)?;
|
||||
memory_file_path.push(String::from(SNAPSHOT_FILENAME));
|
||||
|
||||
let mem_snapshot: MemoryManagerSnapshotData = snapshot
|
||||
.to_versioned_state(MEMORY_MANAGER_SNAPSHOT_ID)
|
||||
.map_err(Error::Restore)?;
|
||||
let mem_snapshot: MemoryManagerSnapshotData =
|
||||
snapshot.to_versioned_state().map_err(Error::Restore)?;
|
||||
|
||||
let mm = MemoryManager::new(
|
||||
vm,
|
||||
@ -2444,7 +2443,6 @@ impl Snapshottable for MemoryManager {
|
||||
self.snapshot_memory_ranges = memory_ranges;
|
||||
|
||||
memory_manager_snapshot.add_data_section(SnapshotDataSection::new_from_versioned_state(
|
||||
MEMORY_MANAGER_SNAPSHOT_ID,
|
||||
&self.snapshot_data(),
|
||||
)?);
|
||||
|
||||
|
@ -4,10 +4,7 @@
|
||||
|
||||
#[cfg(all(target_arch = "x86_64", feature = "guest_debug"))]
|
||||
use crate::coredump::GuestDebuggableError;
|
||||
use crate::{
|
||||
config::VmConfig,
|
||||
vm::{VmSnapshot, VM_SNAPSHOT_ID},
|
||||
};
|
||||
use crate::{config::VmConfig, vm::VmSnapshot};
|
||||
use anyhow::anyhow;
|
||||
use std::fs::File;
|
||||
use std::io::BufReader;
|
||||
@ -71,13 +68,8 @@ pub fn recv_vm_state(source_url: &str) -> std::result::Result<Snapshot, Migratab
|
||||
}
|
||||
|
||||
pub fn get_vm_snapshot(snapshot: &Snapshot) -> std::result::Result<VmSnapshot, MigratableError> {
|
||||
if let Some(vm_section) = snapshot
|
||||
.snapshot_data
|
||||
.get(&format!("{}-section", VM_SNAPSHOT_ID))
|
||||
{
|
||||
return serde_json::from_slice(&vm_section.snapshot).map_err(|e| {
|
||||
MigratableError::Restore(anyhow!("Could not deserialize VM snapshot {}", e))
|
||||
});
|
||||
if let Some(snapshot_data) = snapshot.snapshot_data.as_ref() {
|
||||
return snapshot_data.to_state();
|
||||
}
|
||||
|
||||
Err(MigratableError::Restore(anyhow!(
|
||||
|
@ -2522,7 +2522,6 @@ impl Snapshottable for Vm {
|
||||
|
||||
vm_snapshot.add_snapshot(self.device_manager.lock().unwrap().snapshot()?);
|
||||
vm_snapshot.add_data_section(SnapshotDataSection {
|
||||
id: format!("{}-section", VM_SNAPSHOT_ID),
|
||||
snapshot: vm_snapshot_data,
|
||||
});
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user