1use crate::config::SamplerConfig;
6use anyhow::{Context, Error};
7use argh::FromArgs;
8use fuchsia_component::client::connect_to_protocol;
9use fuchsia_component::server::ServiceFs;
10use fuchsia_inspect::health::Reporter;
11use fuchsia_inspect::{self as inspect};
12use futures::{StreamExt, TryStreamExt};
13use log::{info, warn};
14use sampler_component_config::Config as ComponentConfig;
15use {fidl_fuchsia_hardware_power_statecontrol as reboot, fuchsia_async as fasync};
16
17mod config;
18mod diagnostics;
19mod executor;
20
21pub const PROGRAM_NAME: &str = "sampler";
23
24#[derive(Debug, Default, FromArgs, PartialEq)]
26#[argh(subcommand, name = "sampler")]
27pub struct Args {}
28
29pub async fn main() -> Result<(), Error> {
30 let inspector = inspect::component::inspector();
32 let _inspect_server_task =
33 inspect_runtime::publish(inspector, inspect_runtime::PublishOptions::default());
34
35 let mut service_fs = ServiceFs::new();
36 service_fs.take_and_serve_directory_handle()?;
37 fasync::Task::spawn(async move {
38 service_fs.collect::<()>().await;
39 })
40 .detach();
41
42 inspect::component::health().set_starting_up();
44
45 let sampler_config = SamplerConfig::new(ComponentConfig::take_from_startup_handle())?;
46
47 let (reboot_watcher_client, reboot_watcher_request_stream) =
49 fidl::endpoints::create_request_stream::<reboot::RebootWatcherMarker>();
50
51 {
52 let reboot_watcher_register =
55 connect_to_protocol::<reboot::RebootMethodsWatcherRegisterMarker>()
56 .context("Connect to Reboot watcher register")?;
57
58 reboot_watcher_register
59 .register_watcher(reboot_watcher_client)
60 .await
61 .context("Providing the reboot register with callback channel.")?;
62 }
63
64 let sampler_executor = executor::SamplerExecutor::new(sampler_config).await?;
65
66 let task_canceller = sampler_executor.execute();
69
70 inspect::component::health().set_ok();
71 reboot_watcher(reboot_watcher_request_stream, task_canceller).await;
72 Ok(())
73}
74
75async fn reboot_watcher(
76 mut stream: reboot::RebootWatcherRequestStream,
77 task_canceller: executor::TaskCancellation,
78) {
79 if let Some(reboot::RebootWatcherRequest::OnReboot { options: _, responder }) =
80 stream.try_next().await.unwrap_or_else(|err| {
81 warn!("Reboot callback channel closed: {:?}", err);
84 None
85 })
86 {
87 task_canceller.perform_reboot_cleanup().await;
88
89 responder
91 .send()
92 .unwrap_or_else(|err| warn!("Acking the reboot register failed: {:?}", err));
93
94 info!("Sampler has been halted due to reboot. Goodbye.");
95 } else {
96 task_canceller.run_without_cancellation().await;
100 info!("All Sampler tasks have finished running. Goodbye.");
101 }
102}