_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}