use clap::{Parser, Subcommand};
use std::io::Write;
use std::path::{Path, PathBuf};
use std::process::{self, Command};
#[derive(Parser)]
#[command(name = "xtask", about = "Build and run arceos-loadapp on different architectures")]
struct Cli {
#[command(subcommand)]
command: Cmd,
}
#[derive(Subcommand)]
enum Cmd {
Build {
#[arg(long, default_value = "riscv64")]
arch: String,
},
Run {
#[arg(long, default_value = "riscv64")]
arch: String,
},
}
#[allow(dead_code)]
struct ArchInfo {
target: &'static str,
platform: &'static str,
objcopy_arch: &'static str,
}
fn arch_info(arch: &str) -> ArchInfo {
match arch {
"riscv64" => ArchInfo {
target: "riscv64gc-unknown-none-elf",
platform: "riscv64-qemu-virt",
objcopy_arch: "riscv64",
},
"aarch64" => ArchInfo {
target: "aarch64-unknown-none-softfloat",
platform: "aarch64-qemu-virt",
objcopy_arch: "aarch64",
},
"x86_64" => ArchInfo {
target: "x86_64-unknown-none",
platform: "x86-pc",
objcopy_arch: "x86_64",
},
"loongarch64" => ArchInfo {
target: "loongarch64-unknown-none",
platform: "loongarch64-qemu-virt",
objcopy_arch: "loongarch64",
},
_ => {
eprintln!(
"Error: unsupported architecture '{}'. \
Supported: riscv64, aarch64, x86_64, loongarch64",
arch
);
process::exit(1);
}
}
}
fn project_root() -> PathBuf {
PathBuf::from(env!("CARGO_MANIFEST_DIR"))
}
fn install_config(root: &Path, arch: &str) {
let src = root.join("configs").join(format!("{arch}.toml"));
let dst = root.join(".axconfig.toml");
if !src.exists() {
eprintln!("Error: config file not found: {}", src.display());
process::exit(1);
}
std::fs::copy(&src, &dst).unwrap_or_else(|e| {
eprintln!(
"Error: failed to copy {} -> {}: {}",
src.display(),
dst.display(),
e
);
process::exit(1);
});
println!("Installed config: {} -> .axconfig.toml", src.display());
}
fn create_fat_disk_image(path: &Path) {
const DISK_SIZE: u64 = 64 * 1024 * 1024;
let file = std::fs::OpenOptions::new()
.read(true)
.write(true)
.create(true)
.truncate(true)
.open(path)
.unwrap_or_else(|e| {
eprintln!("Error: failed to create disk image {}: {}", path.display(), e);
process::exit(1);
});
file.set_len(DISK_SIZE).unwrap();
let format_opts = fatfs::FormatVolumeOptions::new().fat_type(fatfs::FatType::Fat32);
fatfs::format_volume(&file, format_opts).unwrap_or_else(|e| {
eprintln!("Error: failed to format FAT32: {}", e);
process::exit(1);
});
{
let fs = fatfs::FileSystem::new(&file, fatfs::FsOptions::new()).unwrap_or_else(|e| {
eprintln!("Error: failed to open FAT filesystem: {}", e);
process::exit(1);
});
let root_dir = fs.root_dir();
root_dir.create_dir("sbin").unwrap_or_else(|e| {
eprintln!("Error: failed to create /sbin directory: {}", e);
process::exit(1);
});
let mut f = root_dir.create_file("sbin/origin.bin").unwrap_or_else(|e| {
eprintln!("Error: failed to create /sbin/origin.bin: {}", e);
process::exit(1);
});
let mut content = [0u8; 64];
for (i, byte) in content.iter_mut().enumerate() {
*byte = (i as u8).wrapping_mul(0x11).wrapping_add(0x10);
}
f.write_all(&content).unwrap();
f.flush().unwrap();
}
println!(
"Created FAT32 disk image: {} ({}MB) with /sbin/origin.bin",
path.display(),
DISK_SIZE / (1024 * 1024)
);
}
fn do_build(root: &Path, info: &ArchInfo) {
let manifest = root.join("Cargo.toml");
let status = Command::new("cargo")
.args([
"build",
"--release",
"--target",
info.target,
"--features",
"axstd",
"--manifest-path",
manifest.to_str().unwrap(),
])
.status()
.expect("failed to execute cargo build");
if !status.success() {
eprintln!("Error: cargo build failed");
process::exit(status.code().unwrap_or(1));
}
}
fn do_objcopy(elf: &Path, bin: &Path, objcopy_arch: &str) {
let status = Command::new("rust-objcopy")
.args([
&format!("--binary-architecture={objcopy_arch}"),
elf.to_str().unwrap(),
"--strip-all",
"-O",
"binary",
bin.to_str().unwrap(),
])
.status()
.expect("failed to execute rust-objcopy (install with: cargo install cargo-binutils)");
if !status.success() {
eprintln!("Error: rust-objcopy failed");
process::exit(status.code().unwrap_or(1));
}
}
fn do_run_qemu(arch: &str, elf: &Path, bin: &Path, disk: &Path) {
let mem = "128M";
let smp = "1";
let qemu = format!("qemu-system-{arch}");
let mut args: Vec<String> = vec![
"-m".into(),
mem.into(),
"-smp".into(),
smp.into(),
"-nographic".into(),
];
match arch {
"riscv64" => {
args.extend([
"-machine".into(),
"virt".into(),
"-bios".into(),
"default".into(),
"-kernel".into(),
bin.to_str().unwrap().into(),
]);
}
"aarch64" => {
args.extend([
"-cpu".into(),
"cortex-a72".into(),
"-machine".into(),
"virt".into(),
"-kernel".into(),
bin.to_str().unwrap().into(),
]);
}
"x86_64" => {
args.extend([
"-machine".into(),
"q35".into(),
"-kernel".into(),
elf.to_str().unwrap().into(),
]);
}
"loongarch64" => {
args.extend([
"-machine".into(),
"virt".into(),
"-kernel".into(),
bin.to_str().unwrap().into(),
]);
}
_ => unreachable!(),
}
args.extend([
"-drive".into(),
format!("file={},format=raw,if=none,id=disk0", disk.display()),
"-device".into(),
"virtio-blk-pci,drive=disk0".into(),
]);
println!("Running: {} {}", qemu, args.join(" "));
let status = Command::new(&qemu)
.args(&args)
.status()
.unwrap_or_else(|e| {
eprintln!("Error: failed to run {}: {}", qemu, e);
process::exit(1);
});
if !status.success() {
process::exit(status.code().unwrap_or(1));
}
}
fn main() {
let cli = Cli::parse();
let root = project_root();
match cli.command {
Cmd::Build { ref arch } => {
let info = arch_info(arch);
install_config(&root, arch);
do_build(&root, &info);
println!("Build complete for {arch} ({})", info.target);
}
Cmd::Run { ref arch } => {
let info = arch_info(arch);
install_config(&root, arch);
do_build(&root, &info);
let elf = root
.join("target")
.join(info.target)
.join("release")
.join("arceos-loadapp");
let bin = elf.with_extension("bin");
let disk = root.join("target").join("disk.img");
create_fat_disk_image(&disk);
if arch != "x86_64" {
do_objcopy(&elf, &bin, info.objcopy_arch);
}
do_run_qemu(arch, &elf, &bin, &disk);
}
}
}