build_rs_guest_arch/lib.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.
//! Centralized build-script logic to support `cfg(guest_arch = ...)` directives
//! throughout the HvLite codebase.
//!
//! By default, `cfg(guest_arch = ...)` will be the same as Rust built-in
//! `cfg(target_arch = ...)`, but this can be overwritten by setting the
//! `OPENVMM_GUEST_TARGET=...` env var at compile time.
//!
//! HvLite code should not be written to assume that `guest_arch` and
//! `target_arch` will always be equal!
//!
//! At some point in the future, HvLite may integrate a full-blown CPU emulator
//! `virt_` backend (akin to what QEMU does), which would allow it to run guest
//! VMs with a _different_ architecture from the host machine.
//!
//! e.g: Aarch64 on x86, x86 on Aarch64, or even exotic things, like RISC-V on
//! x86 (assuming someone cares enough to put in the work there + add
//! appropriate CI coverage!).
#![expect(missing_docs)]
#[derive(Copy, Clone, PartialEq)]
enum GuestArch {
X86_64,
Aarch64,
}
impl GuestArch {
fn all() -> [Self; 2] {
[GuestArch::X86_64, GuestArch::Aarch64]
}
fn as_str(&self) -> &'static str {
match *self {
GuestArch::X86_64 => "x86_64",
GuestArch::Aarch64 => "aarch64",
}
}
fn host_arch() -> Self {
// xtask-fmt allow-target-arch oneoff-guest-arch-impl
std::env::var("CARGO_CFG_TARGET_ARCH")
.unwrap()
.parse()
.unwrap()
}
}
impl std::str::FromStr for GuestArch {
type Err = String;
fn from_str(s: &str) -> Result<Self, Self::Err> {
let arch = match s {
"x86_64" => GuestArch::X86_64,
"aarch64" => GuestArch::Aarch64,
_ => return Err(format!("unknown guest arch '{s}'")),
};
Ok(arch)
}
}
pub fn emit_guest_arch() {
println!("cargo:rerun-if-env-changed=OPENVMM_GUEST_TARGET");
let host_arch = GuestArch::host_arch();
let arch = {
if let Ok(s) = std::env::var("OPENVMM_GUEST_TARGET") {
s.parse().unwrap()
} else {
host_arch
}
};
println!("cargo:rustc-cfg=guest_arch=\"{}\"", arch.as_str());
if host_arch == arch {
println!("cargo:rustc-cfg=guest_is_native");
}
let possible_arches = GuestArch::all()
.map(|a| format!("\"{}\"", a.as_str()))
.join(",");
println!(
"cargo:rustc-check-cfg=cfg(guest_arch, values({}))",
possible_arches
);
println!("cargo:rustc-check-cfg=cfg(guest_is_native)");
}