1#![warn(clippy::unused_async)]
7#![warn(missing_docs, unreachable_patterns, unused)]
8#![recursion_limit = "256"]
9
10mod bindings;
11
12use std::num::NonZeroU8;
13
14use fidl::endpoints::RequestStream as _;
15use fidl_fuchsia_process_lifecycle as fprocess_lifecycle;
16use fuchsia_async as fasync;
17use fuchsia_async::SendExecutorBuilder;
18use fuchsia_component::server::{ServiceFs, ServiceFsDir};
19use futures::{Future, StreamExt as _};
20use log::{error, info};
21
22use bindings::{GlobalConfig, InspectPublisher, InterfaceConfigDefaults, NetstackSeed, Service};
23
24pub fn main() {
26 let config = ns3_config::Config::take_from_startup_handle();
27 let ns3_config::Config {
28 num_threads,
29 debug_logs,
30 opaque_iids,
31 suspend_enabled,
32 sampled_stats_enabled,
33 } = &config;
34 let num_threads = NonZeroU8::new(*num_threads).expect("invalid 0 thread count value");
35 let mut executor = SendExecutorBuilder::new().num_threads(num_threads.get().into()).build();
36
37 let mut log_options = diagnostics_log::PublishOptions::default();
38
39 log_options = log_options.tags(&["netstack"]);
45
46 if *debug_logs {
47 log_options = log_options
50 .minimum_severity(diagnostics_log::Severity::Debug)
51 .listen_for_interest_updates(false);
52 }
53 diagnostics_log::initialize(log_options).expect("failed to initialize log");
54
55 fuchsia_trace_provider::trace_provider_create_with_fdio();
56
57 info!("starting netstack3 with {config:?}");
58
59 let mut fs = ServiceFs::new();
60 let _: &mut ServiceFsDir<'_, _> = fs
61 .dir("svc")
62 .add_proxy_service::<fidl_fuchsia_net_dhcp::ClientProviderMarker, _>()
69 .add_fidl_service(Service::Control)
70 .add_service_connector(Service::DebugDiagnostics)
71 .add_fidl_service(Service::DebugInterfaces)
72 .add_fidl_service(Service::DnsServerWatcher)
73 .add_fidl_service(Service::FilterControl)
74 .add_fidl_service(Service::FilterState)
75 .add_fidl_service(Service::HealthCheck)
76 .add_fidl_service(Service::Interfaces)
77 .add_fidl_service(Service::InterfacesAdmin)
78 .add_fidl_service(Service::MulticastAdminV4)
79 .add_fidl_service(Service::MulticastAdminV6)
80 .add_fidl_service(Service::NdpWatcher)
81 .add_fidl_service(Service::Neighbor)
82 .add_fidl_service(Service::NeighborController)
83 .add_fidl_service(Service::PacketSocket)
84 .add_fidl_service(Service::RawSocket)
85 .add_fidl_service(Service::RootFilter)
86 .add_fidl_service(Service::RootInterfaces)
87 .add_fidl_service(Service::RootRoutesV4)
88 .add_fidl_service(Service::RootRoutesV6)
89 .add_fidl_service(Service::RoutesAdminV4)
90 .add_fidl_service(Service::RoutesAdminV6)
91 .add_fidl_service(Service::RoutesState)
92 .add_fidl_service(Service::RoutesStateV4)
93 .add_fidl_service(Service::RoutesStateV6)
94 .add_fidl_service(Service::RouteTableProviderV4)
95 .add_fidl_service(Service::RouteTableProviderV6)
96 .add_fidl_service(Service::RuleTableV4)
97 .add_fidl_service(Service::RuleTableV6)
98 .add_fidl_service(Service::SettingsControl)
99 .add_fidl_service(Service::SettingsState)
100 .add_fidl_service(Service::Socket)
101 .add_fidl_service(Service::SocketControl)
102 .add_fidl_service(Service::SocketDiagnostics)
103 .add_fidl_service(Service::Stack)
104 .add_fidl_service(Service::WakeGroupProvider);
105
106 let seed = NetstackSeed::new(
107 GlobalConfig {
108 suspend_enabled: *suspend_enabled,
109 sampled_stats_enabled: *sampled_stats_enabled,
110 },
111 &InterfaceConfigDefaults { opaque_iids: *opaque_iids },
112 );
113
114 let inspect_publisher = InspectPublisher::new();
115 inspect_publisher
116 .inspector()
117 .root()
118 .record_child("Config", |config_node| config.record_inspect(config_node));
119
120 let _: &mut ServiceFs<_> = fs.take_and_serve_directory_handle().expect("directory handle");
121
122 let fs = fs.take_until(get_lifecycle_stop_fut());
124
125 executor.run(seed.serve(fs, inspect_publisher))
126}
127
128fn get_lifecycle_stop_fut() -> impl Future<Output = ()> {
131 const LIFECYCLE_HANDLE_ARG: u16 = 0;
134 let handle = fuchsia_runtime::take_startup_handle(fuchsia_runtime::HandleInfo::new(
135 fuchsia_runtime::HandleType::Lifecycle,
136 LIFECYCLE_HANDLE_ARG,
137 ))
138 .expect("missing lifecycle handle");
139
140 async move {
141 let mut request_stream = fprocess_lifecycle::LifecycleRequestStream::from_channel(
142 fasync::Channel::from_channel(handle.into()).into(),
143 );
144 loop {
145 match request_stream.next().await {
146 Some(Ok(fprocess_lifecycle::LifecycleRequest::Stop { control_handle })) => {
147 info!("received shutdown request");
148 std::mem::drop(control_handle);
157 let (inner, _terminated): (_, bool) = request_stream.into_inner();
158 let inner = std::sync::Arc::try_unwrap(inner)
159 .expect("failed to retrieve lifecycle channel");
160 let inner: zx::Channel = inner.into_channel().into_zx_channel();
161 std::mem::forget(inner);
162 break;
163 }
164 Some(Err(e)) => error!("observed error in lifecycle request stream: {e:?}"),
165 None => {
166 error!("lifecycle channel closed");
170 futures::future::pending::<()>().await;
171 }
172 }
173 }
174 }
175}