use std::{
fs,
path::{Path, PathBuf},
};
use anyhow::anyhow;
use ostool::{build::config::Cargo, run::qemu::QemuConfig};
use super::{Axvisor, build};
use crate::{context::ResolvedAxvisorRequest, rootfs};
pub(super) async fn qemu(axvisor: &mut Axvisor, args: super::ArgsQemu) -> anyhow::Result<()> {
let request = axvisor.prepare_request(
(&args.build).into(),
args.qemu_config,
None,
crate::context::SnapshotPersistence::Store,
)?;
axvisor.app.set_debug_mode(request.debug)?;
let cargo = build::load_cargo_config(&request)?;
let explicit_rootfs = args.rootfs.map(|rootfs| {
crate::rootfs::store::resolve_explicit_rootfs(
axvisor.app.workspace_root(),
&request.arch,
rootfs,
)
});
ensure_qemu_rootfs_ready(
&request,
axvisor.app.workspace_root(),
explicit_rootfs.as_deref(),
)
.await?;
let qemu =
load_patched_qemu_config(axvisor, &request, &cargo, explicit_rootfs.as_deref()).await?;
axvisor
.app
.qemu(cargo, request.build_info_path, Some(qemu))
.await
}
pub(super) async fn load_patched_qemu_config(
axvisor: &mut Axvisor,
request: &ResolvedAxvisorRequest,
cargo: &Cargo,
explicit_rootfs: Option<&Path>,
) -> anyhow::Result<QemuConfig> {
let config_path = request.qemu_config.clone().unwrap_or_else(|| {
super::default_qemu_config_template_path(&request.axvisor_dir, &request.arch)
});
let mut qemu = axvisor
.app
.tool_mut()
.read_qemu_config_from_path_for_cargo(cargo, &config_path)
.await?;
patch_qemu_rootfs(
&mut qemu,
request,
axvisor.app.workspace_root(),
explicit_rootfs,
)?;
Ok(qemu)
}
pub(crate) async fn ensure_qemu_rootfs_ready(
request: &ResolvedAxvisorRequest,
workspace_root: &Path,
explicit_rootfs: Option<&Path>,
) -> anyhow::Result<()> {
let rootfs_path = managed_rootfs_path(request, workspace_root, explicit_rootfs)?;
rootfs::store::ensure_optional_managed_rootfs(
workspace_root,
&request.arch,
rootfs_path.as_deref(),
)
.await
}
pub(crate) fn patch_qemu_rootfs(
config: &mut QemuConfig,
request: &ResolvedAxvisorRequest,
workspace_root: &Path,
explicit_rootfs: Option<&Path>,
) -> anyhow::Result<()> {
let rootfs_path = qemu_rootfs_path(request, workspace_root, explicit_rootfs)?;
patch_qemu_rootfs_path(config, &rootfs_path);
Ok(())
}
pub(crate) fn qemu_rootfs_path(
request: &ResolvedAxvisorRequest,
workspace_root: &Path,
explicit_rootfs: Option<&Path>,
) -> anyhow::Result<PathBuf> {
if let Some(explicit) = explicit_rootfs {
return Ok(explicit.to_path_buf());
}
infer_rootfs_path(&request.vmconfigs)?
.map(Ok)
.unwrap_or_else(|| rootfs::store::default_rootfs_path(workspace_root, &request.arch))
}
pub(crate) fn patch_qemu_rootfs_path(config: &mut QemuConfig, rootfs_path: &Path) {
rootfs::qemu::patch_rootfs(
config,
rootfs_path,
rootfs::qemu::RootfsPatchMode::ReplaceDriveOnly,
);
}
pub(crate) fn managed_rootfs_path(
request: &ResolvedAxvisorRequest,
workspace_root: &Path,
explicit_rootfs: Option<&Path>,
) -> anyhow::Result<Option<PathBuf>> {
if let Some(explicit_rootfs) = explicit_rootfs {
if explicit_rootfs.starts_with(rootfs::store::rootfs_dir(workspace_root)) {
return Ok(Some(explicit_rootfs.to_path_buf()));
}
return Ok(None);
}
if infer_rootfs_path(&request.vmconfigs)?.is_none() {
return Ok(Some(rootfs::store::default_rootfs_path(
workspace_root,
&request.arch,
)?));
}
Ok(None)
}
pub(crate) fn infer_rootfs_path(vmconfigs: &[PathBuf]) -> anyhow::Result<Option<PathBuf>> {
for vmconfig in vmconfigs {
let content = fs::read_to_string(vmconfig)
.map_err(|e| anyhow!("failed to read vm config {}: {e}", vmconfig.display()))?;
let value: toml::Value = toml::from_str(&content)
.map_err(|e| anyhow!("failed to parse vm config {}: {e}", vmconfig.display()))?;
let Some(kernel_path) = value
.get("kernel")
.and_then(|kernel| kernel.get("kernel_path"))
.and_then(|path| path.as_str())
else {
continue;
};
let rootfs_path = Path::new(kernel_path)
.parent()
.map(|dir| dir.join("rootfs.img"));
if let Some(rootfs_path) = rootfs_path
&& rootfs_path.exists()
{
return Ok(Some(rootfs_path));
}
}
Ok(None)
}
#[cfg(test)]
mod tests {
use tempfile::tempdir;
use super::*;
fn request(root: &Path, vmconfigs: Vec<PathBuf>) -> ResolvedAxvisorRequest {
ResolvedAxvisorRequest {
package: crate::axvisor::build::AXVISOR_PACKAGE.to_string(),
axvisor_dir: root.join("os/axvisor"),
arch: "aarch64".to_string(),
target: "aarch64-unknown-none-softfloat".to_string(),
plat_dyn: None,
smp: None,
debug: false,
build_info_path: root.join(".build.toml"),
qemu_config: None,
uboot_config: None,
vmconfigs,
}
}
#[test]
fn infer_rootfs_path_uses_vmconfig_kernel_sibling() {
let root = tempdir().unwrap();
let image_dir = root.path().join("image");
fs::create_dir_all(&image_dir).unwrap();
fs::write(image_dir.join("rootfs.img"), b"rootfs").unwrap();
let vmconfig = root.path().join("vm.toml");
fs::write(
&vmconfig,
format!(
r#"
[kernel]
kernel_path = "{}"
"#,
image_dir.join("qemu-aarch64").display()
),
)
.unwrap();
assert_eq!(
infer_rootfs_path(&[vmconfig]).unwrap(),
Some(image_dir.join("rootfs.img"))
);
}
#[test]
fn patch_qemu_rootfs_overrides_rootfs_when_vmconfig_provides_one() {
let root = tempdir().unwrap();
let image_dir = root.path().join("image");
fs::create_dir_all(&image_dir).unwrap();
let rootfs_path = image_dir.join("rootfs.img");
fs::write(&rootfs_path, b"rootfs").unwrap();
let vmconfig = root.path().join("vm.toml");
fs::write(
&vmconfig,
format!(
r#"
[kernel]
kernel_path = "{}"
"#,
image_dir.join("qemu-aarch64").display()
),
)
.unwrap();
let mut qemu = QemuConfig {
args: vec!["id=disk0,if=none,format=raw,file=/old/tmp/rootfs.img".to_string()],
..Default::default()
};
patch_qemu_rootfs(
&mut qemu,
&request(root.path(), vec![vmconfig]),
root.path(),
None,
)
.unwrap();
assert_eq!(
qemu.args,
vec![format!(
"id=disk0,if=none,format=raw,file={}",
rootfs_path.display()
)]
);
}
#[test]
fn patch_qemu_rootfs_uses_unified_rootfs_by_default() {
let root = tempdir().unwrap();
let mut qemu = QemuConfig {
args: vec!["id=disk0,if=none,format=raw,file=/old/tmp/rootfs.img".to_string()],
..Default::default()
};
patch_qemu_rootfs(&mut qemu, &request(root.path(), vec![]), root.path(), None).unwrap();
assert_eq!(
qemu.args,
vec![format!(
"id=disk0,if=none,format=raw,file={}",
root.path()
.join("tmp/axbuild/rootfs/rootfs-aarch64-alpine.img")
.display()
)]
);
}
#[test]
fn patch_qemu_rootfs_inserts_drive_arg_when_template_omits_it() {
let root = tempdir().unwrap();
let mut qemu = QemuConfig {
args: vec![
"-device".to_string(),
"virtio-blk-device,drive=disk0".to_string(),
"-append".to_string(),
"root=/dev/vda rw init=/bin/sh".to_string(),
],
..Default::default()
};
patch_qemu_rootfs(&mut qemu, &request(root.path(), vec![]), root.path(), None).unwrap();
assert_eq!(
qemu.args,
vec![
"-device".to_string(),
"virtio-blk-device,drive=disk0".to_string(),
"-drive".to_string(),
format!(
"id=disk0,if=none,format=raw,file={}",
root.path()
.join("tmp/axbuild/rootfs/rootfs-aarch64-alpine.img")
.display()
),
"-append".to_string(),
"root=/dev/vda rw init=/bin/sh".to_string(),
]
);
}
#[test]
fn managed_rootfs_path_uses_default_unified_rootfs_when_vmconfig_has_no_rootfs() {
let root = tempdir().unwrap();
let vmconfig = root.path().join("vm.toml");
fs::write(
&vmconfig,
r#"
[kernel]
kernel_path = "/tmp/qemu-aarch64"
"#,
)
.unwrap();
assert_eq!(
managed_rootfs_path(&request(root.path(), vec![vmconfig]), root.path(), None).unwrap(),
Some(
root.path()
.join("tmp/axbuild/rootfs/rootfs-aarch64-alpine.img")
)
);
}
#[test]
fn managed_rootfs_path_skips_when_vmconfig_provides_kernel_sibling_rootfs() {
let root = tempdir().unwrap();
let image_dir = root.path().join("image");
fs::create_dir_all(&image_dir).unwrap();
fs::write(image_dir.join("rootfs.img"), b"rootfs").unwrap();
let vmconfig = root.path().join("vm.toml");
fs::write(
&vmconfig,
format!(
r#"
[kernel]
kernel_path = "{}"
"#,
image_dir.join("qemu-aarch64").display()
),
)
.unwrap();
assert_eq!(
managed_rootfs_path(&request(root.path(), vec![vmconfig]), root.path(), None).unwrap(),
None
);
}
#[test]
fn managed_rootfs_path_keeps_explicit_managed_rootfs() {
let root = tempdir().unwrap();
let explicit = root
.path()
.join("tmp/axbuild/rootfs/rootfs-aarch64-debian.img");
assert_eq!(
managed_rootfs_path(
&request(root.path(), vec![]),
root.path(),
Some(explicit.as_path())
)
.unwrap(),
Some(explicit)
);
}
}