Files
asterinas/osdk/src/cli.rs
2024-09-30 16:04:03 +08:00

480 lines
14 KiB
Rust

// SPDX-License-Identifier: MPL-2.0
use std::path::PathBuf;
use clap::{crate_version, Args, Parser, ValueEnum};
use crate::{
arch::Arch,
commands::{
execute_build_command, execute_debug_command, execute_forwarded_command,
execute_new_command, execute_profile_command, execute_run_command, execute_test_command,
},
config::{
manifest::{ProjectType, TomlManifest},
scheme::{BootMethod, BootProtocol},
Config,
},
};
use linux_bzimage_builder::PayloadEncoding;
pub fn main() {
let load_config = |common_args: &CommonArgs| {
let manifest = TomlManifest::load();
let scheme = manifest.get_scheme(common_args.scheme.as_ref());
Config::new(scheme, common_args)
};
let cli = Cli::parse();
let CargoSubcommand::Osdk(osdk_subcommand) = &cli.cargo_subcommand;
match osdk_subcommand {
OsdkSubcommand::New(args) => execute_new_command(args),
OsdkSubcommand::Build(build_args) => {
execute_build_command(&load_config(&build_args.common_args), build_args);
}
OsdkSubcommand::Run(run_args) => {
execute_run_command(
&load_config(&run_args.common_args),
run_args.gdb_server.as_deref(),
);
}
OsdkSubcommand::Debug(debug_args) => {
execute_debug_command(
&load_config(&debug_args.common_args).run.build.profile,
debug_args,
);
}
OsdkSubcommand::Profile(profile_args) => {
execute_profile_command(
&load_config(&profile_args.common_args).run.build.profile,
profile_args,
);
}
OsdkSubcommand::Test(test_args) => {
execute_test_command(&load_config(&test_args.common_args), test_args);
}
OsdkSubcommand::Check(args) => execute_forwarded_command("check", &args.args, true),
OsdkSubcommand::Clippy(args) => execute_forwarded_command("clippy", &args.args, true),
OsdkSubcommand::Doc(args) => execute_forwarded_command("doc", &args.args, false),
}
}
#[derive(Debug, Parser)]
#[clap(display_name = "cargo", bin_name = "cargo")]
/// Project Manager for the crates developed based on frame kernel
pub struct Cli {
#[clap(subcommand)]
cargo_subcommand: CargoSubcommand,
}
#[derive(Debug, Parser)]
enum CargoSubcommand {
#[clap(subcommand, version = crate_version!())]
Osdk(OsdkSubcommand),
}
#[derive(Debug, Parser)]
pub enum OsdkSubcommand {
#[command(about = "Create a new kernel package or library package which depends on OSTD")]
New(NewArgs),
#[command(about = "Compile the project and its dependencies")]
Build(BuildArgs),
#[command(about = "Run the kernel with a VMM")]
Run(RunArgs),
#[command(about = "Debug a remote target via GDB")]
Debug(DebugArgs),
#[command(about = "Profile a remote GDB debug target to collect stack traces for flame graph")]
Profile(ProfileArgs),
#[command(about = "Execute kernel mode unit test by starting a VMM")]
Test(TestArgs),
#[command(about = "Check a local package and all of its dependencies for errors")]
Check(ForwardedArguments),
#[command(about = "Checks a package to catch common mistakes and improve your Rust code")]
Clippy(ForwardedArguments),
#[command(about = "Build a package's documentation")]
Doc(ForwardedArguments),
}
#[derive(Debug, Parser)]
pub struct ForwardedArguments {
#[arg(
help = "The full set of Cargo arguments",
trailing_var_arg = true,
allow_hyphen_values = true
)]
pub args: Vec<String>,
}
#[derive(Debug, Parser)]
pub struct NewArgs {
#[arg(
id = "type",
long = "type",
short = 't',
default_value = "library",
help = "The type of the project to create",
conflicts_with_all = ["kernel", "library"],
)]
pub type_: ProjectType,
#[arg(
long,
help = "Create a kernel package",
conflicts_with_all = ["library", "type"],
)]
pub kernel: bool,
#[arg(
long,
alias = "lib",
help = "Create a library package",
conflicts_with_all = ["kernel", "type"],
)]
pub library: bool,
#[arg(name = "name", required = true)]
pub crate_name: String,
}
impl NewArgs {
pub fn project_type(&self) -> ProjectType {
if self.kernel {
ProjectType::Kernel
} else if self.library {
ProjectType::Library
} else {
self.type_
}
}
}
#[derive(Debug, Parser)]
pub struct BuildArgs {
#[arg(
long = "for-test",
help = "Build for running unit tests",
default_value_t
)]
pub for_test: bool,
#[arg(
long = "output",
short = 'o',
help = "Output directory for all generated artifacts",
value_name = "DIR"
)]
pub output: Option<PathBuf>,
#[command(flatten)]
pub common_args: CommonArgs,
}
#[derive(Debug, Parser)]
pub struct RunArgs {
#[arg(
long = "gdb-server",
help = "Enable the QEMU GDB server for debugging\n\
This option supports an additional comma separated configuration list:\n\t \
addr=ADDR: the network or unix socket address on which the GDB server listens, \
`.osdk-gdb-socket` by default;\n\t \
wait-client: let the GDB server wait for the GDB client before execution;\n\t \
vscode: generate a '.vscode/launch.json' for debugging with Visual Studio Code.",
value_name = "[addr=ADDR][,wait-client][,vscode]",
default_missing_value = ""
)]
pub gdb_server: Option<String>,
#[command(flatten)]
pub common_args: CommonArgs,
}
#[derive(Debug, Parser)]
pub struct DebugArgs {
#[arg(
long,
help = "Specify the address of the remote target",
default_value = ".osdk-gdb-socket"
)]
pub remote: String,
#[command(flatten)]
pub common_args: CommonArgs,
}
#[derive(Debug, Parser)]
pub struct ProfileArgs {
#[arg(
long,
help = "Specify the address of the remote target",
default_value = ".osdk-gdb-socket"
)]
pub remote: String,
#[arg(long, help = "The number of samples to collect", default_value = "200")]
pub samples: usize,
#[arg(
long,
help = "The interval between samples in seconds",
default_value = "0.1"
)]
pub interval: f64,
#[arg(
long,
help = "Parse a collected JSON profile file into other formats",
value_name = "PATH",
conflicts_with = "samples",
conflicts_with = "interval"
)]
pub parse: Option<PathBuf>,
#[command(flatten)]
pub out_args: DebugProfileOutArgs,
#[command(flatten)]
pub common_args: CommonArgs,
}
#[derive(Clone, Copy, Debug, ValueEnum)]
pub enum ProfileFormat {
/// The raw stack trace log parsed from GDB in JSON
Json,
/// The folded stack trace for generating a flame graph later using
/// [the original tool](https://github.com/brendangregg/FlameGraph)
Folded,
/// A SVG flame graph
FlameGraph,
}
impl ProfileFormat {
pub fn file_extension(&self) -> &'static str {
match self {
ProfileFormat::Json => "json",
ProfileFormat::Folded => "folded",
ProfileFormat::FlameGraph => "svg",
}
}
}
#[derive(Debug, Parser)]
pub struct DebugProfileOutArgs {
#[arg(long, help = "The output format for the profile data")]
format: Option<ProfileFormat>,
#[arg(
long,
help = "The mask of the CPU to generate traces for in the output profile data",
default_value_t = u128::MAX
)]
pub cpu_mask: u128,
#[arg(
long,
help = "The path to the output profile data file",
value_name = "PATH"
)]
output: Option<PathBuf>,
}
impl DebugProfileOutArgs {
/// Get the output format for the profile data.
///
/// If the user does not specify the format, it will be inferred from the
/// output file extension. If the output file does not have an extension,
/// the default format is flame graph.
pub fn format(&self) -> ProfileFormat {
self.format.unwrap_or_else(|| {
if self.output.is_some() {
match self.output.as_ref().unwrap().extension() {
Some(ext) if ext == "folded" => ProfileFormat::Folded,
Some(ext) if ext == "json" => ProfileFormat::Json,
Some(ext) if ext == "svg" => ProfileFormat::FlameGraph,
_ => ProfileFormat::FlameGraph,
}
} else {
ProfileFormat::FlameGraph
}
})
}
/// Get the output path for the profile data.
///
/// If the user does not specify the output path, it will be generated from
/// the current time stamp and the format. The caller can provide a hint
/// output path to the file to override the file name.
pub fn output_path(&self, hint: Option<&PathBuf>) -> PathBuf {
self.output.clone().unwrap_or_else(|| {
use chrono::{offset::Local, DateTime};
let file_stem = if let Some(hint) = hint {
format!(
"{}",
hint.parent()
.unwrap()
.join(hint.file_stem().unwrap())
.display()
)
} else {
let crate_name = crate::util::get_current_crate_info().name;
let time_stamp = std::time::SystemTime::now();
let time_stamp: DateTime<Local> = time_stamp.into();
let time_stamp = time_stamp.format("%H%M%S");
format!("{}-profile-{}", crate_name, time_stamp)
};
PathBuf::from(format!("{}.{}", file_stem, self.format().file_extension()))
})
}
}
#[derive(Debug, Parser)]
pub struct TestArgs {
#[arg(
name = "TESTNAME",
help = "Only run tests containing this string in their names"
)]
pub test_name: Option<String>,
#[command(flatten)]
pub common_args: CommonArgs,
}
#[derive(Debug, Args, Default, Clone, Eq, PartialEq, Serialize, Deserialize)]
pub struct CargoArgs {
#[arg(
long,
help = "The Cargo build profile (built-in candidates are 'dev', 'release', 'test' and 'bench')",
conflicts_with = "release",
global = true
)]
pub profile: Option<String>,
#[arg(
long,
help = "Build artifacts in release mode",
conflicts_with = "profile",
global = true
)]
pub release: bool,
#[arg(
long,
value_name = "FEATURES",
help = "List of features to activate",
value_delimiter = ',',
num_args = 1..,
global = true,
)]
pub features: Vec<String>,
#[arg(long, help = "Do not activate the `default` features", global = true)]
pub no_default_features: bool,
#[arg(
long = "config",
help = "Override a configuration value",
value_name = "KEY=VALUE",
global = true
)]
pub override_configs: Vec<String>,
}
impl CargoArgs {
pub fn profile(&self) -> Option<String> {
if self.release {
Some("release".to_owned())
} else {
self.profile.clone()
}
}
}
#[derive(Debug, Args)]
/// Common args used for build, run, test and debug subcommand
pub struct CommonArgs {
#[command(flatten)]
pub build_args: CargoArgs,
#[arg(
long = "linux-x86-legacy-boot",
help = "Enable legacy 32-bit boot support for the Linux x86 boot protocol",
global = true
)]
pub linux_x86_legacy_boot: bool,
#[arg(
long = "strip-elf",
help = "Strip the built kernel ELF file for a smaller size",
global = true
)]
pub strip_elf: bool,
#[arg(
long = "target-arch",
value_name = "ARCH",
help = "The architecture to build for",
global = true
)]
pub target_arch: Option<Arch>,
#[arg(
long = "scheme",
help = "Select the specific configuration scheme provided in the OSDK manifest",
value_name = "SCHEME",
global = true
)]
pub scheme: Option<String>,
#[arg(
long = "kcmd-args",
require_equals = true,
help = "Extra or overriding command line arguments for guest kernel",
value_name = "ARGS",
global = true
)]
pub kcmd_args: Vec<String>,
#[arg(
long = "init-args",
require_equals = true,
help = "Extra command line arguments for init process",
value_name = "ARGS",
global = true
)]
pub init_args: Vec<String>,
#[arg(long, help = "Path of initramfs", value_name = "PATH", global = true)]
pub initramfs: Option<PathBuf>,
#[arg(
long = "boot-method",
help = "Loader for booting the kernel",
value_name = "BOOTMETHOD",
global = true
)]
pub boot_method: Option<BootMethod>,
#[arg(
long = "bootdev-append-options",
help = "Additional QEMU `-drive` options for the boot device",
value_name = "OPTIONS",
global = true
)]
pub bootdev_append_options: Option<String>,
#[arg(
long = "display-grub-menu",
help = "Display the GRUB menu if booting with GRUB",
global = true
)]
pub display_grub_menu: bool,
#[arg(
long = "grub-mkrescue",
help = "Path of grub-mkrescue",
value_name = "PATH",
global = true
)]
pub grub_mkrescue: Option<PathBuf>,
#[arg(
long = "grub-boot-protocol",
help = "Protocol for booting the kernel",
value_name = "BOOT_PROTOCOL",
global = true
)]
pub grub_boot_protocol: Option<BootProtocol>,
#[arg(
long = "qemu-exe",
help = "The QEMU executable file",
value_name = "FILE",
global = true
)]
pub qemu_exe: Option<PathBuf>,
#[arg(
long = "qemu-args",
require_equals = true,
help = "Extra arguments or overriding arguments for running QEMU",
value_name = "ARGS",
global = true
)]
pub qemu_args: Vec<String>,
#[arg(
long = "encoding",
help = "Denote the encoding format for kernel self-decompression",
value_name = "FORMAT",
global = true
)]
pub encoding: Option<PayloadEncoding>,
}