_wlansoftmac_c_rustc_static/lib.rs
1// Copyright 2022 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5//! C bindings for wlansoftmac-rust crate.
6#![deny(unsafe_op_in_unsafe_fn)]
7
8use diagnostics_log::PublishOptions;
9use fuchsia_async::LocalExecutor;
10use log::info;
11use std::ffi::c_void;
12use std::sync::Once;
13use wlan_ffi_transport::completers::Completer;
14use wlan_ffi_transport::{EthernetRx, FfiEthernetRx, FfiWlanTx, WlanTx};
15use wlan_mlme::device::Device;
16use {fidl_fuchsia_wlan_softmac as fidl_softmac, wlan_trace as wtrace};
17
18use fdf::{DispatcherBuilder, OnDispatcher};
19
20static LOGGER_ONCE: Once = Once::new();
21
22/// Start and run a bridged wlansoftmac driver hosting an MLME server and an SME server.
23///
24/// The driver is "bridged" in the sense that it requires a bridge to a Fuchsia driver to
25/// communicate with other Fuchsia drivers over the FDF transport. After the bridged
26/// driver starts successfully, `run_start_completer` will be called with `start_completer`.
27/// If the bridged driver does not start successfully, this function will return a
28/// non-`ZX_OK` status.
29///
30/// This function returns `ZX_OK` only if shutdown completes successfully. A successful
31/// shutdown only occurs if the bridged driver receives a
32/// `fuchsia.wlan.softmac/WlanSoftmacIfcBridge.StopBridgedDriver` message and the subsequent
33/// teardown of the hosted server succeeds.
34///
35/// In all other scenarios, e.g., failure during startup, while running, or during shutdown,
36/// this function will return a non-`ZX_OK` value.
37///
38/// # Safety
39///
40/// This function is unsafe for the following reasons:
41///
42/// - This function cannot guarantee `run_start_completer` is thread-safe.
43/// - This function cannot guarantee `start_completer` points to a valid object when
44/// `run_start_completer` is called.
45/// - This function cannot guarantee `run_shutdown_completer` is thread-safe.
46/// - This function cannot guarantee `shutdown_completer` points to a valid object when
47/// `run_shutdown_completer` is called.
48/// - This function cannot guarantee `wlan_softmac_bridge_client_handle` is a valid handle.
49///
50/// By calling this function, the caller promises the following:
51///
52/// - The `run_start_completer` function is thread-safe.
53/// - The `start_completer` pointer will point to a valid object at least until
54/// `run_start_completer` is called.
55/// - The `run_shutdown_completer` function is thread-safe.
56/// - The `shutdown_completer` pointer will point to a valid object at least until
57/// `run_shutdown_completer` is called.
58/// - The `wlan_softmac_bridge_client_handle` is a valid handle.
59#[no_mangle]
60pub unsafe extern "C" fn start_bridged_wlansoftmac(
61 start_completer: *mut c_void,
62 run_start_completer: unsafe extern "C" fn(
63 start_completer: *mut c_void,
64 status: zx::sys::zx_status_t,
65 ),
66 shutdown_completer: *mut c_void,
67 run_shutdown_completer: unsafe extern "C" fn(
68 shutdown_completer: *mut c_void,
69 status: zx::sys::zx_status_t,
70 ),
71 ethernet_rx: FfiEthernetRx,
72 wlan_tx: FfiWlanTx,
73 wlan_softmac_bridge_client_handle: zx::sys::zx_handle_t,
74) -> zx::sys::zx_status_t {
75 // The Fuchsia syslog must not be initialized from Rust more than once per process. In the case
76 // of MLME, that means we can only call it once for both the client and ap modules. Ensure this
77 // by using a shared `Once::call_once()`.
78 LOGGER_ONCE.call_once(|| {
79 if log::log_enabled!(log::Level::Info) {
80 // Assuming logger is already initialized if log level is enabled.
81 return;
82 }
83 // Initialize logging with a tag that can be used to filter for forwarding to console
84 diagnostics_log::initialize_sync(
85 PublishOptions::default()
86 .tags(&["wlan"])
87 .enable_metatag(diagnostics_log::Metatag::Target),
88 );
89 });
90
91 // Safety: The provided closure is safe to send to another thread because start_completer
92 // and run_start_completer are thread-safe.
93 let start_completer = unsafe {
94 Completer::new_unchecked(move |status| {
95 // Safety: This is safe because the caller of this function promised
96 // `run_start_completer` is thread-safe and `start_completer` is valid until
97 // its called.
98 run_start_completer(start_completer, status);
99 })
100 };
101
102 let dispatcher = match DispatcherBuilder::new()
103 .name("bridged-wlansoftmac")
104 .allow_thread_blocking()
105 .shutdown_observer(|_dispatcher| {
106 wtrace::duration!(c"bridged-wlansoftmac dispatcher shutdown_handler");
107 info!("Completed bridged-wlansoftmac dispatcher shutdown");
108 })
109 // Create a released dispatcher to defer destruction until driver
110 // shutdown. The framework will shutdown the dispatcher
111 // despite its release here because the framework tagged this
112 // driver as the owner of the dispatcher upon creation. The
113 // framework always automatically shuts down all dispatchers
114 // owned by a driver during the driver's shut down.
115 .create_released()
116 {
117 Ok(dispatcher) => dispatcher,
118 Err(status) => {
119 info!("Failed to create dispatcher for MLME: {}", status);
120 start_completer.reply(status.into());
121 return status.into_raw();
122 }
123 };
124
125 let ethernet_rx = EthernetRx::new(ethernet_rx);
126 let wlan_tx = WlanTx::new(wlan_tx);
127
128 // Safety: The provided closure is safe to send to another thread because shutdown_completer
129 // and run_shutdown_completer are thread-safe.
130 let shutdown_completer = unsafe {
131 Completer::new_unchecked(move |status| {
132 // Safety: This is safe because the caller of this function promised
133 // `run_shutdown_completer` is thread-safe and `shutdown_completer` is valid until
134 // its called.
135 run_shutdown_completer(shutdown_completer, status);
136 })
137 };
138
139 let task = dispatcher.spawn_task(async move {
140 wtrace::duration!(c"Rust MLME dispatcher");
141
142 let mut executor = LocalExecutor::new();
143
144 let wlan_softmac_bridge_proxy = {
145 // Safety: This is safe because the caller promises `wlan_softmac_bridge_client_handle`
146 // is a valid handle.
147 let handle = unsafe { fidl::Handle::from_raw(wlan_softmac_bridge_client_handle) };
148 let channel = fidl::Channel::from(handle);
149 let channel = fidl::AsyncChannel::from_channel(channel);
150 fidl_softmac::WlanSoftmacBridgeProxy::new(channel)
151 };
152 let device = Device::new(wlan_softmac_bridge_proxy, ethernet_rx, wlan_tx);
153
154 let result =
155 executor.run_singlethreaded(wlansoftmac_rust::start_and_serve(start_completer, device));
156 shutdown_completer.reply(result);
157 });
158
159 zx::Status::from(task).into_raw()
160}