_wlan_fullmac_mlme_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
5use fidl_fuchsia_wlan_fullmac as fidl_fullmac;
6use log::error;
7use std::ffi::c_void;
8use wlan_ffi_transport::completers::Completer;
9use wlan_fullmac_mlme::device::FullmacDevice;
10use wlan_fullmac_mlme::{start_and_serve_on_separate_thread, FullmacMlmeHandle};
11
12/// Starts and runs the FullMAC MLME and SME futures on a separate thread. MLME will call the given
13/// `run_shutdown_completer` when it exits.
14///
15/// # Safety
16///
17/// This function is unsafe for the following reasons:
18///
19/// - This function cannot guarantee `fullmac_client_end_handle` is a valid handle.
20/// - This function cannot guarantee `run_shutdown_completer` is thread-safe.
21/// - This function cannot guarantee `shutdown_completer` points to a valid object when
22/// `run_shutdown_completer` is called.
23///
24/// By calling this function, the caller promises the following:
25///
26/// - The `fullmac_client_end_handle` is a valid handle.
27/// - The `run_shutdown_completer` function is thread-safe.
28/// - The `shutdown_completer` pointer will point to a valid object at least until
29/// `run_shutdown_completer` is called.
30///
31/// It is additionally the caller's responsibility to deallocate the returned FullmacMlmeHandle.
32/// The caller promises that they will eventually call `delete_fullmac_mlme` on the returned
33/// FullmacMlmeHandle.
34#[no_mangle]
35pub unsafe extern "C" fn start_fullmac_mlme(
36 fullmac_client_end_handle: zx::sys::zx_handle_t,
37 shutdown_completer: *mut c_void,
38 run_shutdown_completer: unsafe extern "C" fn(
39 shutdown_completer: *mut c_void,
40 status: zx::sys::zx_status_t,
41 ),
42) -> *mut FullmacMlmeHandle {
43 let fullmac_impl_sync_proxy = {
44 // Safety: This is safe because the caller promises `fullmac_client_end_handle`
45 // is a valid handle.
46 let handle = unsafe { fidl::Handle::from_raw(fullmac_client_end_handle) };
47 let channel = fidl::Channel::from(handle);
48 fidl_fullmac::WlanFullmacImpl_SynchronousProxy::new(channel)
49 };
50
51 // Safety: The provided closure is safe to send to another thread because shutdown_completer
52 // and run_shutdown_completer are thread-safe.
53 let shutdown_completer = unsafe {
54 Completer::new_unchecked(move |status| {
55 // Safety: This is safe because the caller of this function promised
56 // `run_shutdown_completer` is thread-safe and `shutdown_completer` is valid until
57 // its called.
58 run_shutdown_completer(shutdown_completer, status);
59 })
60 };
61
62 let device = FullmacDevice::new(fullmac_impl_sync_proxy);
63 match start_and_serve_on_separate_thread(device, shutdown_completer) {
64 Ok(mlme) => Box::into_raw(Box::new(mlme)),
65 Err(e) => {
66 error!("Failed to start FullMAC MLME: {}", e);
67 std::ptr::null_mut()
68 }
69 }
70}
71
72/// Request that the FullMAC MLME stops. This is non-blocking.
73///
74/// It is assumed that |mlme| is valid (i.e., the user has not called
75/// |delete_fullmac_mlme_handle| yet.
76///
77/// This should be synchronized with calls to |start_fullmac_mlme| and
78/// |delete_fullmac_mlme_handle|.
79///
80/// TODO(https://fxbug.dev/368323681): Consider replacing |stop_fullmac_mlme| and
81/// |delete_fullmac_mlme| with an internal FIDL protocol.
82#[no_mangle]
83pub extern "C" fn stop_fullmac_mlme(mlme: &mut FullmacMlmeHandle) {
84 mlme.request_stop();
85}
86
87/// Takes ownership of and deallocates the passed FullmacMlmeHandle.
88///
89/// If |mlme| is non-null, this will join the FullMAC MLME thread and block the calling thread
90/// until FullMAC MLME has exited.
91///
92/// If the user has not called |stop_fullmac_mlme| before |delete_fullmac_mlme_handle|, this will
93/// request that the FullMAC MLME thread stop before joining thread to avoid blocking forever.
94///
95/// TODO(https://fxbug.dev/368323681): Consider replacing |stop_fullmac_mlme| and
96/// |delete_fullmac_mlme| with an internal FIDL protocol.
97///
98/// # Safety
99///
100/// This fn accepts a raw pointer that is held by the FFI caller as a handle to
101/// the MLME. This API is fundamentally unsafe, and relies on the caller to
102/// pass the correct pointer and make no further calls on it later.
103#[no_mangle]
104pub unsafe extern "C" fn delete_fullmac_mlme_handle(mlme: *mut FullmacMlmeHandle) {
105 if !mlme.is_null() {
106 let mlme = Box::from_raw(mlme);
107 mlme.delete();
108 }
109}