1use crate::logs::{LoggerError, LoggerStream, create_log_stream, create_std_combined_log_stream};
8use anyhow::Error;
9use cm_types::NamespacePath;
10use fidl_fuchsia_component::IntrospectorMarker;
11use fidl_fuchsia_io as fio;
12use fidl_fuchsia_process as fproc;
13use fuchsia_component::client::connect_to_protocol;
14use fuchsia_component::directory::AsRefDirectory;
15use fuchsia_runtime as runtime;
16use namespace::Namespace;
17use runtime::{HandleInfo, HandleType};
18use thiserror::Error;
19use zx::{Process, Rights, Task};
20
21pub static UTC_CLOCK_BASIC_RIGHTS: std::sync::LazyLock<zx::Rights> =
34 std::sync::LazyLock::new(|| {
35 Rights::DUPLICATE
36 | Rights::READ
37 | Rights::WAIT
38 | Rights::TRANSFER
39 | Rights::MAP
40 | Rights::INSPECT
41 });
42
43#[derive(Debug, Error)]
45pub enum LaunchError {
46 #[error("{:?}", _0)]
47 Logger(#[from] LoggerError),
48
49 #[error("Error connecting to launcher: {:?}", _0)]
50 Launcher(Error),
51
52 #[error("{:?}", _0)]
53 LoadInfo(runner::component::LaunchError),
54
55 #[error("Error launching process: {:?}", _0)]
56 LaunchCall(fidl::Error),
57
58 #[error("Error launching process: {:?}", _0)]
59 ProcessLaunch(zx::Status),
60
61 #[error("Error duplicating vDSO: {:?}", _0)]
62 DuplicateVdso(zx::Status),
63
64 #[error("Error launching process: {:?}", _0)]
65 Fidl(#[from] fidl::Error),
66
67 #[error("Error launching process, cannot create socket {:?}", _0)]
68 CreateSocket(zx::Status),
69
70 #[error("Error cloning UTC clock: {:?}", _0)]
71 UtcClock(zx::Status),
72
73 #[error("unexpected error")]
74 UnExpectedError,
75}
76
77pub struct LaunchProcessArgs<'a> {
79 pub bin_path: &'a str,
81 pub process_name: &'a str,
84 pub job: Option<zx::Job>,
86 pub ns: Namespace,
88 pub args: Option<Vec<String>>,
90 pub name_infos: Option<Vec<fproc::NameInfo>>,
92 pub environs: Option<Vec<String>>,
94 pub handle_infos: Option<Vec<fproc::HandleInfo>>,
97 pub loader_proxy_chan: Option<zx::Channel>,
99 pub executable_vmo: Option<zx::Vmo>,
101 pub options: zx::ProcessOptions,
103 pub config_vmo: Option<zx::Vmo>,
105 pub component_instance: Option<fidl::Event>,
107 pub url: Option<String>,
109}
110
111pub async fn launch_process(
113 args: LaunchProcessArgs<'_>,
114) -> Result<(Process, ScopedJob, LoggerStream), LaunchError> {
115 let launcher = connect_to_protocol::<fproc::LauncherMarker>().map_err(LaunchError::Launcher)?;
116 let (logger, stdout_handle, stderr_handle) =
117 create_std_combined_log_stream().map_err(LaunchError::Logger)?;
118 let (process, job) = launch_process_impl(args, launcher, stdout_handle, stderr_handle).await?;
119 Ok((process, job, logger))
120}
121
122pub async fn launch_process_with_separate_std_handles(
125 args: LaunchProcessArgs<'_>,
126) -> Result<(Process, ScopedJob, LoggerStream, LoggerStream), LaunchError> {
127 let launcher = connect_to_protocol::<fproc::LauncherMarker>().map_err(LaunchError::Launcher)?;
128 let (stdout_logger, stdout_handle) = create_log_stream().map_err(LaunchError::Logger)?;
129 let (stderr_logger, stderr_handle) = create_log_stream().map_err(LaunchError::Logger)?;
130 let (process, job) = launch_process_impl(args, launcher, stdout_handle, stderr_handle).await?;
131 Ok((process, job, stdout_logger, stderr_logger))
132}
133
134async fn launch_process_impl(
135 args: LaunchProcessArgs<'_>,
136 launcher: fproc::LauncherProxy,
137 stdout_handle: zx::NullableHandle,
138 stderr_handle: zx::NullableHandle,
139) -> Result<(Process, ScopedJob), LaunchError> {
140 const STDOUT: u16 = 1;
141 const STDERR: u16 = 2;
142
143 let mut handle_infos = args.handle_infos.unwrap_or(vec![]);
144
145 handle_infos.push(fproc::HandleInfo {
146 handle: stdout_handle,
147 id: HandleInfo::new(HandleType::FileDescriptor, STDOUT).as_raw(),
148 });
149
150 handle_infos.push(fproc::HandleInfo {
151 handle: stderr_handle,
152 id: HandleInfo::new(HandleType::FileDescriptor, STDERR).as_raw(),
153 });
154
155 handle_infos.push(fproc::HandleInfo {
156 handle: runtime::duplicate_utc_clock_handle(*UTC_CLOCK_BASIC_RIGHTS)
157 .map_err(LaunchError::UtcClock)?
158 .into_handle(),
159 id: HandleInfo::new(HandleType::ClockUtc, 0).as_raw(),
160 });
161
162 if let Some(config_vmo) = args.config_vmo {
163 handle_infos.push(fproc::HandleInfo {
164 handle: config_vmo.into_handle(),
165 id: HandleInfo::new(HandleType::ComponentConfigVmo, 0).as_raw(),
166 });
167 }
168
169 if let Some(svc_dir) = args.ns.get(&NamespacePath::new("/svc").unwrap()) {
170 let (client, server) = zx::Channel::create();
171 svc_dir
172 .as_ref_directory()
173 .open("fuchsia.logger.LogSink", fio::Flags::PROTOCOL_SERVICE, server.into())
174 .expect("open LogSink for test");
175 handle_infos.push(fproc::HandleInfo {
176 handle: client.into(),
177 id: runtime::HandleInfo::new(runtime::HandleType::LogSink, 0).as_raw(),
178 });
179 }
180
181 let LaunchProcessArgs {
182 bin_path,
183 process_name,
184 args,
185 options,
186 ns,
187 job,
188 name_infos,
189 environs,
190 loader_proxy_chan,
191 executable_vmo,
192 component_instance,
193 url,
194 ..
195 } = args;
196 let launch_info =
198 runner::component::configure_launcher(runner::component::LauncherConfigArgs {
199 bin_path,
200 name: process_name,
201 args,
202 options,
203 ns,
204 job,
205 handle_infos: Some(handle_infos),
206 name_infos,
207 environs,
208 launcher: &launcher,
209 loader_proxy_chan,
210 executable_vmo,
211 })
212 .await
213 .map_err(LaunchError::LoadInfo)?;
214
215 let component_job = launch_info
216 .job
217 .duplicate_handle(zx::Rights::SAME_RIGHTS)
218 .expect("handle duplication failed!");
219
220 let (status, process) = launcher.launch(launch_info).await.map_err(LaunchError::LaunchCall)?;
221
222 let status = zx::Status::from_raw(status);
223 if status != zx::Status::OK {
224 return Err(LaunchError::ProcessLaunch(status));
225 }
226
227 let process = process.ok_or_else(|| LaunchError::UnExpectedError)?;
228
229 trace_component_start(&process, component_instance, url).await;
230
231 Ok((process, ScopedJob::new(component_job)))
232}
233
234async fn trace_component_start(
238 process: &Process,
239 component_instance: Option<fidl::Event>,
240 url: Option<String>,
241) {
242 if fuchsia_trace::category_enabled(c"component:start") {
243 let pid = process.koid().unwrap().raw_koid();
244 let moniker = match component_instance {
245 None => "Missing component instance".to_string(),
246 Some(component_instance) => match connect_to_protocol::<IntrospectorMarker>() {
247 Ok(introspector) => {
248 let component_instance =
249 component_instance.duplicate_handle(zx::Rights::SAME_RIGHTS).unwrap();
250 match introspector.get_moniker(component_instance).await {
251 Ok(Ok(moniker)) => moniker,
252 Ok(Err(e)) => {
253 format!("Couldn't get moniker: {e:?}")
254 }
255 Err(e) => {
256 format!("Couldn't get the moniker: {e:?}")
257 }
258 }
259 }
260 Err(e) => {
261 format!("Couldn't get introspector: {e:?}")
262 }
263 },
264 };
265 let url = url.unwrap_or_else(|| "Missing URL".to_string());
266 fuchsia_trace::instant!(
267 c"component:start",
268 c"-test-",
271 fuchsia_trace::Scope::Thread,
272 "moniker" => format!("{}", moniker).as_str(),
273 "url" => url.as_str(),
274 "pid" => pid
275 );
276 }
277}
278
279pub struct ScopedJob {
281 pub object: Option<zx::Job>,
282}
283
284impl ScopedJob {
285 pub fn new(job: zx::Job) -> Self {
286 Self { object: Some(job) }
287 }
288
289 pub fn take(mut self) -> zx::Job {
291 self.object.take().unwrap()
292 }
293}
294
295impl Drop for ScopedJob {
296 fn drop(&mut self) {
297 if let Some(job) = self.object.take() {
298 job.kill().ok();
299 }
300 }
301}
302
303#[cfg(test)]
304mod tests {
305 use super::*;
306 use fidl::endpoints::{ClientEnd, Proxy, create_proxy_and_stream};
307 use fidl_fuchsia_component_runner as fcrunner;
308 use fidl_fuchsia_io as fio;
309 use fuchsia_async as fasync;
310 use fuchsia_runtime::{job_default, process_self, swap_utc_clock_handle};
311 use futures::prelude::*;
312
313 #[test]
314 fn scoped_job_works() {
315 let new_job = job_default().create_child_job().unwrap();
316 let job_dup = new_job.duplicate_handle(zx::Rights::SAME_RIGHTS).unwrap();
317
318 let _child_job = new_job.create_child_job().unwrap();
320
321 let info = job_dup.info().unwrap();
323 assert!(!info.exited);
324 {
325 let _job_about_to_die = ScopedJob::new(new_job);
326 }
327
328 let info = job_dup.info().unwrap();
330 assert!(info.exited);
331 }
332
333 #[test]
334 fn scoped_job_take_works() {
335 let new_job = job_default().create_child_job().unwrap();
336 let raw_handle = new_job.raw_handle();
337
338 let scoped = ScopedJob::new(new_job);
339
340 let ret_job = scoped.take();
341
342 assert_eq!(ret_job.raw_handle(), raw_handle);
344 }
345
346 #[fasync::run_singlethreaded(test)]
347 #[ignore] async fn utc_clock_is_cloned() {
349 let clock = fuchsia_runtime::UtcClock::create(zx::ClockOpts::MONOTONIC, None)
350 .expect("failed to create clock");
351 let expected_clock_koid = clock.koid().expect("failed to get clock koid");
352
353 let _ = swap_utc_clock_handle(clock).expect("failed to swap clocks");
356
357 let pkg = fuchsia_fs::directory::open_in_namespace(
360 "/pkg",
361 fio::PERM_READABLE | fio::PERM_EXECUTABLE,
362 )
363 .expect("failed to open pkg");
364 let args = LaunchProcessArgs {
365 bin_path: "bin/test_runners_lib_lib_test", environs: None,
367 args: None,
368 job: None,
369 process_name: "foo",
370 name_infos: None,
371 handle_infos: None,
372 ns: vec![fcrunner::ComponentNamespaceEntry {
373 path: Some("/pkg".into()),
374 directory: Some(ClientEnd::new(pkg.into_channel().unwrap().into_zx_channel())),
375 ..Default::default()
376 }]
377 .try_into()
378 .unwrap(),
379 loader_proxy_chan: None,
380 executable_vmo: None,
381 options: zx::ProcessOptions::empty(),
382 config_vmo: None,
383 url: None,
384 component_instance: None,
385 };
386 let (mock_proxy, mut mock_stream) = create_proxy_and_stream::<fproc::LauncherMarker>();
387 let mock_fut = async move {
388 let mut all_handles = vec![];
389 while let Some(request) =
390 mock_stream.try_next().await.expect("failed to get next message")
391 {
392 match request {
393 fproc::LauncherRequest::AddHandles { handles, .. } => {
394 all_handles.extend(handles);
395 }
396 fproc::LauncherRequest::Launch { responder, .. } => {
397 responder
398 .send(
399 zx::Status::OK.into_raw(),
400 Some(
401 process_self()
402 .duplicate_handle(zx::Rights::SAME_RIGHTS)
403 .expect("failed to duplicate process handle"),
404 ),
405 )
406 .expect("failed to send reply");
407 }
408 _ => {}
409 }
410 }
411 return all_handles;
412 };
413 let (_logger, stdout_handle, stderr_handle) =
414 create_std_combined_log_stream().map_err(LaunchError::Logger).unwrap();
415 let client_fut = async move {
416 let _ = launch_process_impl(args, mock_proxy, stdout_handle, stderr_handle)
417 .await
418 .expect("failed to launch process");
419 };
420
421 let (all_handles, ()) = futures::future::join(mock_fut, client_fut).await;
422 let clock_id = HandleInfo::new(HandleType::ClockUtc, 0).as_raw();
423
424 let utc_clock_handle = all_handles
425 .into_iter()
426 .find_map(
427 |hi: fproc::HandleInfo| if hi.id == clock_id { Some(hi.handle) } else { None },
428 )
429 .expect("UTC clock handle");
430 let clock_koid = utc_clock_handle.koid().expect("failed to get koid");
431 assert_eq!(expected_clock_koid, clock_koid);
432 }
433}