elf_runner/
runtime_dir.rs1use fidl::endpoints::ServerEnd;
6use fidl_fuchsia_io as fio;
7use std::sync::Arc;
8use vfs::directory::entry_container::Directory;
9use vfs::directory::helper::DirectlyMutable;
10use vfs::directory::immutable::simple as pfs;
11use vfs::execution_scope::ExecutionScope;
12use vfs::file::vmo::read_only;
13use vfs::object_request::ToObjectRequest as _;
14use vfs::tree_builder::TreeBuilder;
15
16pub struct RuntimeDirectory(Arc<pfs::Simple>);
18
19impl RuntimeDirectory {
20 pub fn add_process_id(&self, process_id: u64) {
21 self.elf_dir()
22 .add_entry("process_id", read_only(process_id.to_string()))
23 .expect("failed to add process_id");
24 }
25
26 pub fn add_process_start_time(&self, process_start_time: i64) {
27 self.elf_dir()
28 .add_entry("process_start_time", read_only(process_start_time.to_string()))
29 .expect("failed to add process_start_time");
30 }
31
32 pub fn add_process_start_time_utc_estimate(&self, process_start_time_utc_estimate: String) {
33 self.elf_dir()
34 .add_entry(
35 "process_start_time_utc_estimate",
36 read_only(process_start_time_utc_estimate),
37 )
38 .expect("failed to add process_start_time_utc_estimate");
39 }
40
41 #[cfg(test)]
43 pub fn empty() -> Self {
44 let mut empty = TreeBuilder::empty_dir();
45 empty.add_empty_dir(["elf"]).expect("failed to add elf directory");
46 RuntimeDirectory(empty.build())
47 }
48
49 fn elf_dir(&self) -> Arc<pfs::Simple> {
50 self.0
51 .get_entry("elf")
52 .expect("elf directory should be present")
53 .into_any()
54 .downcast::<pfs::Simple>()
55 .expect("could not downcast elf to a directory")
56 }
57}
58
59pub struct RuntimeDirBuilder {
60 args: Vec<String>,
61 job_id: Option<u64>,
62 server_end: ServerEnd<fio::NodeMarker>,
63}
64
65impl RuntimeDirBuilder {
66 pub fn new(server_end: ServerEnd<fio::DirectoryMarker>) -> Self {
67 let server_end = ServerEnd::<fio::NodeMarker>::new(server_end.into_channel());
69 Self { args: vec![], job_id: None, server_end }
70 }
71
72 pub fn args(mut self, args: Vec<String>) -> Self {
73 self.args = args;
74 self
75 }
76
77 pub fn job_id(mut self, job_id: u64) -> Self {
78 self.job_id = Some(job_id);
79 self
80 }
81
82 pub fn serve(mut self) -> RuntimeDirectory {
83 let mut runtime_tree_builder = TreeBuilder::empty_dir();
94 let mut count: u32 = 0;
95 for arg in self.args.drain(..) {
96 runtime_tree_builder
97 .add_entry(["args", &count.to_string()], read_only(arg))
98 .expect("Failed to add arg to runtime directory");
99 count += 1;
100 }
101
102 runtime_tree_builder.add_empty_dir(["elf"]).expect("failed to add elf directory");
104
105 if let Some(job_id) = self.job_id {
106 runtime_tree_builder
107 .add_entry(["elf", "job_id"], read_only(job_id.to_string()))
108 .expect("Failed to add job_id to runtime/elf directory");
109 }
110
111 let runtime_directory = runtime_tree_builder.build();
112
113 const SERVE_FLAGS: fio::Flags = fio::PERM_READABLE.union(fio::PERM_WRITABLE);
114
115 let object_request = SERVE_FLAGS.to_object_request(self.server_end.into_channel());
117 let dir = runtime_directory.clone();
118 object_request.handle(|request| {
119 dir.open3(ExecutionScope::new(), vfs::Path::dot(), SERVE_FLAGS, request)
120 });
121 RuntimeDirectory(runtime_directory)
122 }
123}