use clap::{Parser, Subcommand};
use std::path::{Path, PathBuf};
use std::process::{self, Command};
#[derive(Parser)]
#[command(name = "xtask", about = "Build and run arceos-helloworld 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 do_build(root: &Path, info: &ArchInfo) {
let manifest = root.join("Cargo.toml");
let ax_config = root.join(".axconfig.toml");
let status = Command::new("cargo")
.args([
"build",
"--release",
"--target",
info.target,
"--manifest-path",
manifest.to_str().unwrap(),
])
.env("AX_CONFIG_PATH", ax_config.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) {
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!(),
}
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-helloworld");
let bin = elf.with_extension("bin");
if arch != "x86_64" {
do_objcopy(&elf, &bin, info.objcopy_arch);
}
do_run_qemu(arch, &elf, &bin);
}
}
}