From 72c3ea0620a621779187f378b12f3379f4122adc Mon Sep 17 00:00:00 2001 From: Sven Friedrich Date: Tue, 18 Jun 2024 14:29:29 +0200 Subject: [PATCH] feat: add stdio redirection example --- Cargo.lock | 15 +++-- Cargo.toml | 7 +- core/Cargo.toml | 2 +- examples/redirect_stdio/Cargo.toml | 13 ++++ examples/redirect_stdio/redirect_stdio.yaml | 12 ++++ examples/redirect_stdio/src/main.rs | 75 +++++++++++++++++++++ flake.nix | 20 +++++- hypervisor/Cargo.toml | 3 +- partition/Cargo.toml | 2 +- 9 files changed, 137 insertions(+), 12 deletions(-) create mode 100644 examples/redirect_stdio/Cargo.toml create mode 100644 examples/redirect_stdio/redirect_stdio.yaml create mode 100644 examples/redirect_stdio/src/main.rs diff --git a/Cargo.lock b/Cargo.lock index ee58997..6238c9c 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -555,10 +555,6 @@ dependencies = [ "a653rs-postcard", "humantime", "log", - "memmap2", - "nix", - "once_cell", - "procfs", "serde", ] @@ -1064,6 +1060,17 @@ dependencies = [ "getrandom", ] +[[package]] +name = "redirect_stdio" +version = "0.1.0" +dependencies = [ + "a653rs", + "a653rs-linux", + "anyhow", + "log", + "nix", +] + [[package]] name = "regex" version = "1.10.5" diff --git a/Cargo.toml b/Cargo.toml index 658501b..638a398 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -15,15 +15,18 @@ members = [ "examples/ping/client", "examples/ping/server", - "examples/dev_random", - "examples/ping_queue/client", "examples/ping_queue/server", + + "examples/dev_random", + + "examples/redirect_stdio" ] [workspace.dependencies] a653rs = "0.6" a653rs-linux.path = "partition" +anyhow = "1.0" log = "0" nix = { version = "0.29", features = ["socket", "process", "fs", "uio", "signal", "user", "mount", "event", "sched"] } memmap2 = "0.9" diff --git a/core/Cargo.toml b/core/Cargo.toml index 4c462fa..1218b68 100644 --- a/core/Cargo.toml +++ b/core/Cargo.toml @@ -14,8 +14,8 @@ memmap2.workspace = true procfs.workspace = true polling.workspace = true itertools.workspace = true +anyhow.workspace = true -anyhow = "1.0" log = "0" walkdir = "2.3" serde = { version = "1.0", features = ["derive"] } diff --git a/examples/redirect_stdio/Cargo.toml b/examples/redirect_stdio/Cargo.toml new file mode 100644 index 0000000..76cf060 --- /dev/null +++ b/examples/redirect_stdio/Cargo.toml @@ -0,0 +1,13 @@ +[package] +name = "redirect_stdio" +version = "0.1.0" +edition = "2021" + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[dependencies] +a653rs = { workspace = true, features = ["macros"] } +a653rs-linux.workspace = true +log.workspace = true +nix.workspace = true +anyhow.workspace = true diff --git a/examples/redirect_stdio/redirect_stdio.yaml b/examples/redirect_stdio/redirect_stdio.yaml new file mode 100644 index 0000000..3a2fbec --- /dev/null +++ b/examples/redirect_stdio/redirect_stdio.yaml @@ -0,0 +1,12 @@ +major_frame: 1s +partitions: + - id: 0 + name: partition_0 + duration: 1s + offset: 0ms + period: 1s + image: ./target/x86_64-unknown-linux-musl/release/redirect_stdio + mounts: + - [ /tmp/redirect_stdio/stdout, /stdout ] + - [ /tmp/redirect_stdio/stderr, /stderr ] + - [ /tmp/redirect_stdio/stdin, /stdin ] diff --git a/examples/redirect_stdio/src/main.rs b/examples/redirect_stdio/src/main.rs new file mode 100644 index 0000000..5bceeee --- /dev/null +++ b/examples/redirect_stdio/src/main.rs @@ -0,0 +1,75 @@ +use std::fs::OpenOptions; +use std::os::fd::AsRawFd; +use std::path::Path; + +use a653rs::partition; +use a653rs::prelude::PartitionExt; +use a653rs_linux::partition::ApexLogger; +use anyhow::Result; +use log::LevelFilter; + +fn replace_stdio>(stdio: T, new: U, write: bool) -> Result<()> { + let new = OpenOptions::new() + .write(write) + .read(!write) + .truncate(write) + .open(new)?; + nix::unistd::dup2(new.as_raw_fd(), stdio.as_raw_fd())?; + Ok(()) +} + +fn main() { + replace_stdio(std::io::stdin(), "/stdin", false).unwrap(); + replace_stdio(std::io::stdout(), "/stdout", true).unwrap(); + replace_stdio(std::io::stderr(), "/stderr", true).unwrap(); + + ApexLogger::install_panic_hook(); + ApexLogger::install_logger(LevelFilter::Trace).unwrap(); + + redirect_stdio::Partition.run() +} + +#[partition(a653rs_linux::partition::ApexLinuxPartition)] +mod redirect_stdio { + use log::info; + use std::{io::BufRead, time::Duration}; + + #[start(cold)] + fn cold_start(mut ctx: start::Context) { + // create and start an aperiodic process + ctx.create_process_0().unwrap().start().unwrap(); + } + + // do the same as a cold_start + #[start(warm)] + fn warm_start(ctx: start::Context) { + cold_start(ctx); + } + + // this aperiodic process opens /dev/random and reads some random bytes from it + #[aperiodic( + time_capacity = "Infinite", + stack_size = "8KB", + base_priority = 1, + deadline = "Soft" + )] + fn process_0(_: process_0::Context) { + info!("started process_0"); + + println!("Start reading stdin to stdout"); + let stdin = std::io::stdin(); + for line in stdin.lock().lines() { + println!("{}", line.unwrap()) + } + println!("Finished reading stdin to stdout"); + + eprintln!("Error was encountered: None"); + eprintln!("But it was printed to stderr"); + + // TODO wait for https://github.com/DLR-FT/a653rs/issues/22 to be fixed + // Hypervisor::set_partition_mode(OperatingMode::Idle); + loop { + std::thread::sleep(Duration::from_secs(10)) + } + } +} diff --git a/flake.nix b/flake.nix index ee61b24..27c932b 100644 --- a/flake.nix +++ b/flake.nix @@ -71,6 +71,18 @@ name = "hello_part_no_macros"; partitions = [ "hello_part_no_macros" ]; } + { + name = "redirect_stdio"; + partitions = [ "redirect_stdio" ]; + preRun = '' + tmpdir=$(mktemp -d -t redirect_stdio.XXXXXXXXXX) + cp examples/redirect_stdio/redirect_stdio.yaml $tmpdir/config.yaml + sed -i 's@/tmp/redirect_stdio@'"$tmpdir"'@g' $tmpdir/config.yaml + touch $tmpdir/std{out,err} + echo $'hello\nworld!\n' > $tmpdir/stdin + ''; + yaml = "$tmpdir/config.yaml"; + } { name = "fuel_tank"; partitions = [ "fuel_tank_simulation" "fuel_tank_controller" ]; @@ -103,7 +115,7 @@ cargoTestOptions = x: x ++ [ "--package" pname ]; } // env; } // (builtins.listToAttrs (builtins.map - ({ name, partitions }: { + ({ name, partitions, ... }: { name = "example-${name}"; value = naersk-lib.buildPackage rec { @@ -181,7 +193,7 @@ inherit (nixpkgs.lib) flatten; in flatten (map - ({ name, partitions }: [ + ({ name, partitions, preRun ? "", yaml ? "examples/${name}/${name}.yaml" }: [ { name = "run-example-${name}"; command = '' @@ -192,8 +204,10 @@ # prepend PATH so that partition images can be found PATH="target/${rust-target}/release:$PATH" + ${preRun} + # (build &) run hypervisor - RUST_LOG=''${RUST_LOG:=trace} cargo run --package a653rs-linux-hypervisor --release -- examples/${name}/${name}.yaml $@ + RUST_LOG=''${RUST_LOG:=trace} cargo run --package a653rs-linux-hypervisor --release -- ${yaml} $@ ''; help = "Run the ${name} example, consisting of the partitions: ${concatStringsSep "," partitions}"; category = "example"; diff --git a/hypervisor/Cargo.toml b/hypervisor/Cargo.toml index 16aac51..3232ccf 100644 --- a/hypervisor/Cargo.toml +++ b/hypervisor/Cargo.toml @@ -21,7 +21,8 @@ polling.workspace = true itertools.workspace = true once_cell.workspace = true bytesize.workspace = true -anyhow = "1.0" +anyhow.workspace = true + tempfile = "3.3" clone3 = "0.2" serde = { version = "1.0", features = ["derive"] } diff --git a/partition/Cargo.toml b/partition/Cargo.toml index 454059a..43ca1fd 100644 --- a/partition/Cargo.toml +++ b/partition/Cargo.toml @@ -18,9 +18,9 @@ memmap2.workspace = true procfs.workspace = true polling.workspace = true once_cell.workspace = true +anyhow.workspace = true lazy_static = "1.4" -anyhow = "1.0" log.workspace = true tinyvec = "1.6" oneshot = "0.1.6"