vmm: hold vm after vm restore success

Vmm may hold an err vm if vm restore failed. We will encounter a
boot err at this time. This patch modifies the assignment timing
after successful restore.

Signed-off-by: Songqian Li <sionli@tencent.com>
This commit is contained in:
Songqian Li 2024-12-05 16:06:36 +08:00
parent 0c2f2d3ec1
commit 48cdd98a0b
2 changed files with 39 additions and 45 deletions

View File

@ -23,7 +23,7 @@ use libc::{cfmakeraw, isatty, tcgetattr, tcsetattr, termios, TCSANOW};
use thiserror::Error; use thiserror::Error;
use crate::sigwinch_listener::listen_for_sigwinch_on_tty; use crate::sigwinch_listener::listen_for_sigwinch_on_tty;
use crate::vm_config::ConsoleOutputMode; use crate::vm_config::{ConsoleOutputMode, VmConfig};
use crate::Vmm; use crate::Vmm;
const TIOCSPTLCK: libc::c_int = 0x4004_5431; const TIOCSPTLCK: libc::c_int = 0x4004_5431;
@ -176,8 +176,15 @@ fn dup_stdout() -> vmm_sys_util::errno::Result<File> {
Ok(unsafe { File::from_raw_fd(stdout) }) Ok(unsafe { File::from_raw_fd(stdout) })
} }
pub(crate) fn pre_create_console_devices(vmm: &mut Vmm) -> ConsoleDeviceResult<ConsoleInfo> { pub(crate) fn pre_create_console_devices(
let vm_config = vmm.vm_config.as_mut().unwrap().clone(); vmm: &mut Vmm,
new_vm_config: Option<Arc<Mutex<VmConfig>>>,
) -> ConsoleDeviceResult<ConsoleInfo> {
let vm_config = if let Some(vm_config) = new_vm_config {
vm_config
} else {
vmm.vm_config.clone().unwrap()
};
let mut vmconfig = vm_config.lock().unwrap(); let mut vmconfig = vm_config.lock().unwrap();
let console_info = ConsoleInfo { let console_info = ConsoleInfo {

View File

@ -763,7 +763,7 @@ impl Vmm {
let config = vm_migration_config.vm_config.clone(); let config = vm_migration_config.vm_config.clone();
self.vm_config = Some(vm_migration_config.vm_config); self.vm_config = Some(vm_migration_config.vm_config);
self.console_info = Some(pre_create_console_devices(self).map_err(|e| { self.console_info = Some(pre_create_console_devices(self, None).map_err(|e| {
MigratableError::MigrateReceive(anyhow!("Error creating console devices: {:?}", e)) MigratableError::MigrateReceive(anyhow!("Error creating console devices: {:?}", e))
})?); })?);
@ -1244,21 +1244,17 @@ impl RequestHandler for Vmm {
// We only store the passed VM config. // We only store the passed VM config.
// The VM will be created when being asked to boot it. // The VM will be created when being asked to boot it.
if self.vm_config.is_none() { if self.vm_config.is_none() {
self.vm_config = Some(Arc::new(Mutex::new(*config))); let vm_config = Arc::new(Mutex::new(*config));
self.console_info =
Some(pre_create_console_devices(self).map_err(VmError::CreateConsoleDevices)?);
if self self.console_info = Some(
.vm_config pre_create_console_devices(self, Some(vm_config.clone()))
.as_ref() .map_err(VmError::CreateConsoleDevices)?,
.unwrap() );
.lock()
.unwrap() if vm_config.lock().unwrap().landlock_enable {
.landlock_enable apply_landlock(vm_config.clone()).map_err(VmError::ApplyLandlock)?;
{
apply_landlock(self.vm_config.as_ref().unwrap().clone())
.map_err(VmError::ApplyLandlock)?;
} }
self.vm_config = Some(vm_config);
Ok(()) Ok(())
} else { } else {
Err(VmError::VmAlreadyCreated) Err(VmError::VmAlreadyCreated)
@ -1278,8 +1274,10 @@ impl RequestHandler for Vmm {
// console_info is set to None in vm_shutdown. re-populate here if empty // console_info is set to None in vm_shutdown. re-populate here if empty
if self.console_info.is_none() { if self.console_info.is_none() {
self.console_info = self.console_info = Some(
Some(pre_create_console_devices(self).map_err(VmError::CreateConsoleDevices)?); pre_create_console_devices(self, None)
.map_err(VmError::CreateConsoleDevices)?,
);
} }
// Create a new VM if we don't have one yet. // Create a new VM if we don't have one yet.
@ -1404,12 +1402,12 @@ impl RequestHandler for Vmm {
self.vm_check_cpuid_compatibility(&vm_config, &vm_snapshot.common_cpuid) self.vm_check_cpuid_compatibility(&vm_config, &vm_snapshot.common_cpuid)
.map_err(VmError::Restore)?; .map_err(VmError::Restore)?;
self.vm_config = Some(Arc::clone(&vm_config));
// console_info is set to None in vm_snapshot. re-populate here if empty // console_info is set to None in vm_snapshot. re-populate here if empty
if self.console_info.is_none() { if self.console_info.is_none() {
self.console_info = self.console_info = Some(
Some(pre_create_console_devices(self).map_err(VmError::CreateConsoleDevices)?); pre_create_console_devices(self, Some(vm_config.clone()))
.map_err(VmError::CreateConsoleDevices)?,
);
} }
let exit_evt = self.exit_evt.try_clone().map_err(VmError::EventFdClone)?; let exit_evt = self.exit_evt.try_clone().map_err(VmError::EventFdClone)?;
@ -1424,8 +1422,8 @@ impl RequestHandler for Vmm {
.try_clone() .try_clone()
.map_err(VmError::EventFdClone)?; .map_err(VmError::EventFdClone)?;
let vm = Vm::new( let mut vm = Vm::new(
vm_config, vm_config.clone(),
exit_evt, exit_evt,
reset_evt, reset_evt,
#[cfg(feature = "guest_debug")] #[cfg(feature = "guest_debug")]
@ -1440,26 +1438,15 @@ impl RequestHandler for Vmm {
Some(source_url), Some(source_url),
Some(restore_cfg.prefault), Some(restore_cfg.prefault),
)?; )?;
if vm_config.lock().unwrap().landlock_enable {
apply_landlock(vm_config.clone()).map_err(VmError::ApplyLandlock)?;
}
vm.restore()?;
self.vm = Some(vm); self.vm = Some(vm);
self.vm_config = Some(vm_config);
if self Ok(())
.vm_config
.as_ref()
.unwrap()
.lock()
.unwrap()
.landlock_enable
{
apply_landlock(self.vm_config.as_ref().unwrap().clone())
.map_err(VmError::ApplyLandlock)?;
}
// Now we can restore the rest of the VM.
if let Some(ref mut vm) = self.vm {
vm.restore()
} else {
Err(VmError::VmNotCreated)
}
} }
#[cfg(all(target_arch = "x86_64", feature = "guest_debug"))] #[cfg(all(target_arch = "x86_64", feature = "guest_debug"))]
@ -1523,7 +1510,7 @@ impl RequestHandler for Vmm {
} }
self.console_info = self.console_info =
Some(pre_create_console_devices(self).map_err(VmError::CreateConsoleDevices)?); Some(pre_create_console_devices(self, None).map_err(VmError::CreateConsoleDevices)?);
// Then we create the new VM // Then we create the new VM
let mut vm = Vm::new( let mut vm = Vm::new(