openthread/ot/
cli.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 crate::prelude_internal::*;
6use std::os::raw::c_int;
7
8/// Methods from the [OpenThread "CLI" Module](https://openthread.io/reference/group/api-cli).
9pub trait Cli {
10    /// Functional equivalent of [`otsys::otCliInputLine`](crate::otsys::otCliInputLine).
11    fn cli_input_line(&self, line: &CStr);
12
13    /// Functional equivalent of [`otsys::otCliInit`](crate::otsys::otCliInit).
14    fn cli_init<'a, F>(&self, output_callback: F)
15    where
16        F: FnMut(&CStr) + 'a;
17}
18
19impl<T: Cli + ot::Boxable> Cli for ot::Box<T> {
20    fn cli_input_line(&self, line: &CStr) {
21        self.as_ref().cli_input_line(line);
22    }
23
24    fn cli_init<'a, F>(&self, output_callback: F)
25    where
26        F: FnMut(&CStr) + 'a,
27    {
28        self.as_ref().cli_init(output_callback);
29    }
30}
31
32impl Cli for Instance {
33    fn cli_input_line(&self, line: &CStr) {
34        unsafe { otCliInputLine(line.as_ptr() as *mut c_char) }
35    }
36
37    fn cli_init<'a, F>(&self, f: F)
38    where
39        F: FnMut(&CStr) + 'a,
40    {
41        unsafe extern "C" fn _ot_cli_output_callback(
42            context: *mut ::std::os::raw::c_void,
43            line: *const c_char,
44            _: *mut otsys::__va_list_tag,
45        ) -> c_int {
46            let line = CStr::from_ptr(line);
47
48            trace!("_ot_cli_output_callback: {:?}", line);
49
50            // Reconstitute a reference to our instance
51            let instance = Instance::ref_from_ot_ptr(context as *mut otInstance).unwrap();
52
53            // Get a reference to our instance backing.
54            let backing = instance.borrow_backing();
55
56            // Remove our line sender from the cell in the backing.
57            // We will replace it once we are done.
58            let mut sender_option = backing.cli_output_fn.replace(None);
59
60            if let Some(sender) = sender_option.as_mut() {
61                sender(line);
62            } else {
63                info!("_ot_cli_output_callback: Nothing to handle CLI output");
64            }
65
66            // Put the sender back into the backing so that we can
67            // use it again later.
68            if let Some(new_sender) = backing.cli_output_fn.replace(sender_option) {
69                // In this case someone called the init function while
70                // they were in the callback. We want to restore that
71                // callback.
72                backing.cli_output_fn.set(Some(new_sender));
73            }
74
75            line.to_bytes().len().try_into().unwrap()
76        }
77
78        // Put our sender closure in a box.
79        // This will go into the backing.
80        let fn_box = Box::new(f) as Box<dyn FnMut(&CStr) + 'a>;
81
82        // Prepare a function pointer for our callback closure.
83        let cb: otCliOutputCallback = Some(_ot_cli_output_callback);
84
85        // Go ahead and get our context pointer ready for the callback.
86        // In this case our context pointer is the `otInstance` pointer.
87        let fn_ptr = self.as_ot_ptr() as *mut ::std::os::raw::c_void;
88
89        unsafe {
90            // Make sure our object eventually gets cleaned up.
91            // Here we must also transmute our closure to have a 'static
92            // lifetime.
93            // SAFETY: We need to do this because the borrow checker
94            //         cannot infer the proper lifetime for the
95            //         singleton instance backing, but this is guaranteed
96            //         by the API.
97            let prev_callback =
98                self.borrow_backing().cli_output_fn.replace(std::mem::transmute::<
99                    Option<Box<dyn FnMut(&'_ CStr) + 'a>>,
100                    Option<Box<dyn FnMut(&'_ CStr) + 'static>>,
101                >(Some(fn_box)));
102
103            // Check to see if there was a previous callback registered.
104            // We only want to call `otCliInit()` if there was no
105            // previously registered callback.
106            if prev_callback.is_none() {
107                // SAFETY: This must only be called once.
108                otCliInit(self.as_ot_ptr(), cb, fn_ptr);
109            }
110        }
111    }
112}