1use crate::prelude_internal::*;
6use std::time::Duration;
7
8pub trait Link {
12 fn get_channel(&self) -> ChannelIndex;
14
15 fn set_channel(&self, index: ChannelIndex) -> Result;
17
18 fn link_get_counters(&self) -> &MacCounters;
20
21 fn get_extended_address(&self) -> &ExtAddress;
24
25 fn get_factory_assigned_ieee_eui_64(&self) -> ExtAddress;
28
29 fn get_pan_id(&self) -> PanId;
31
32 fn set_pan_id(&self, pan_id: PanId) -> Result;
34
35 fn get_short_address(&self) -> ShortAddress;
37
38 fn link_is_enabled(&self) -> bool;
40
41 fn link_set_enabled(&self, enabled: bool) -> Result;
43
44 fn link_is_promiscuous(&self) -> bool;
46
47 fn link_set_promiscuous(&self, promiscuous: bool) -> Result;
50
51 fn is_energy_scan_in_progress(&self) -> bool;
54
55 fn is_active_scan_in_progress(&self) -> bool;
58
59 fn get_supported_channel_mask(&self) -> ot::ChannelMask;
62
63 fn start_active_scan<'a, F>(&self, channels: ChannelMask, dwell: Duration, f: F) -> Result
69 where
70 F: FnMut(Option<&ActiveScanResult>) + 'a;
71
72 fn start_energy_scan<'a, F>(&self, channels: ChannelMask, dwell: Duration, f: F) -> Result
78 where
79 F: FnMut(Option<&EnergyScanResult>) + 'a;
80}
81
82impl<T: Link + Boxable> Link for ot::Box<T> {
83 fn get_channel(&self) -> ChannelIndex {
84 self.as_ref().get_channel()
85 }
86
87 fn set_channel(&self, index: ChannelIndex) -> Result {
88 self.as_ref().set_channel(index)
89 }
90
91 fn link_get_counters(&self) -> &MacCounters {
92 self.as_ref().link_get_counters()
93 }
94
95 fn get_extended_address(&self) -> &ExtAddress {
96 self.as_ref().get_extended_address()
97 }
98 fn get_factory_assigned_ieee_eui_64(&self) -> ExtAddress {
99 self.as_ref().get_factory_assigned_ieee_eui_64()
100 }
101
102 fn get_pan_id(&self) -> PanId {
103 self.as_ref().get_pan_id()
104 }
105 fn set_pan_id(&self, pan_id: PanId) -> Result<()> {
106 self.as_ref().set_pan_id(pan_id)
107 }
108
109 fn get_short_address(&self) -> ShortAddress {
110 self.as_ref().get_short_address()
111 }
112
113 fn link_is_enabled(&self) -> bool {
114 self.as_ref().link_is_enabled()
115 }
116 fn link_set_enabled(&self, enabled: bool) -> Result {
117 self.as_ref().link_set_enabled(enabled)
118 }
119
120 fn link_is_promiscuous(&self) -> bool {
121 self.as_ref().link_is_promiscuous()
122 }
123 fn link_set_promiscuous(&self, promiscuous: bool) -> Result {
124 self.as_ref().link_set_promiscuous(promiscuous)
125 }
126
127 fn is_energy_scan_in_progress(&self) -> bool {
128 self.as_ref().is_energy_scan_in_progress()
129 }
130 fn is_active_scan_in_progress(&self) -> bool {
131 self.as_ref().is_active_scan_in_progress()
132 }
133
134 fn get_supported_channel_mask(&self) -> ChannelMask {
135 self.as_ref().get_supported_channel_mask()
136 }
137
138 fn start_active_scan<'a, F>(&self, channels: ChannelMask, dwell: Duration, f: F) -> Result
139 where
140 F: FnMut(Option<&ActiveScanResult>) + 'a,
141 {
142 self.as_ref().start_active_scan(channels, dwell, f)
143 }
144
145 fn start_energy_scan<'a, F>(&self, channels: ChannelMask, dwell: Duration, f: F) -> Result
146 where
147 F: FnMut(Option<&EnergyScanResult>) + 'a,
148 {
149 self.as_ref().start_energy_scan(channels, dwell, f)
150 }
151}
152
153impl Link for Instance {
154 fn get_channel(&self) -> ChannelIndex {
155 unsafe { otLinkGetChannel(self.as_ot_ptr()) }
156 }
157
158 fn set_channel(&self, index: ChannelIndex) -> Result {
159 Error::from(unsafe { otLinkSetChannel(self.as_ot_ptr(), index) }).into()
160 }
161
162 fn link_get_counters(&self) -> &MacCounters {
163 unsafe {
164 let mac_counters = otLinkGetCounters(self.as_ot_ptr());
165 MacCounters::ref_from_ot_ptr(mac_counters).unwrap()
166 }
167 }
168
169 fn get_extended_address(&self) -> &ExtAddress {
170 unsafe {
171 let ext_addr = otLinkGetExtendedAddress(self.as_ot_ptr());
172 ExtAddress::ref_from_ot_ptr(ext_addr).unwrap()
173 }
174 }
175
176 fn get_factory_assigned_ieee_eui_64(&self) -> ExtAddress {
177 unsafe {
178 let mut ext_addr = ExtAddress::default();
179 otLinkGetFactoryAssignedIeeeEui64(self.as_ot_ptr(), ext_addr.as_ot_mut_ptr());
180 ext_addr
181 }
182 }
183
184 fn get_pan_id(&self) -> PanId {
185 unsafe { otLinkGetPanId(self.as_ot_ptr()) }
186 }
187
188 fn set_pan_id(&self, pan_id: PanId) -> Result {
189 Error::from(unsafe { otLinkSetPanId(self.as_ot_ptr(), pan_id) }).into()
190 }
191
192 fn get_short_address(&self) -> ShortAddress {
193 unsafe { otLinkGetShortAddress(self.as_ot_ptr()) }
194 }
195
196 fn link_is_enabled(&self) -> bool {
197 unsafe { otLinkIsEnabled(self.as_ot_ptr()) }
198 }
199
200 fn link_set_enabled(&self, enabled: bool) -> Result {
201 Error::from(unsafe { otLinkSetEnabled(self.as_ot_ptr(), enabled) }).into()
202 }
203
204 fn link_is_promiscuous(&self) -> bool {
205 unsafe { otLinkIsPromiscuous(self.as_ot_ptr()) }
206 }
207
208 fn link_set_promiscuous(&self, promiscuous: bool) -> Result {
209 Error::from(unsafe { otLinkSetPromiscuous(self.as_ot_ptr(), promiscuous) }).into()
210 }
211
212 fn is_energy_scan_in_progress(&self) -> bool {
213 unsafe { otLinkIsEnergyScanInProgress(self.as_ot_ptr()) }
214 }
215
216 fn is_active_scan_in_progress(&self) -> bool {
217 unsafe { otLinkIsActiveScanInProgress(self.as_ot_ptr()) }
218 }
219
220 fn get_supported_channel_mask(&self) -> ChannelMask {
221 let mask_u32 = unsafe { otLinkGetSupportedChannelMask(self.as_ot_ptr()) };
222 mask_u32.into()
223 }
224
225 fn start_active_scan<'a, F>(&self, channels: ChannelMask, dwell: Duration, f: F) -> Result
226 where
227 F: FnMut(Option<&ActiveScanResult>) + 'a,
228 {
229 unsafe extern "C" fn _ot_handle_active_scan_result<
230 'a,
231 F: FnMut(Option<&ActiveScanResult>) + 'a,
232 >(
233 result: *mut otActiveScanResult,
234 context: *mut ::std::os::raw::c_void,
235 ) {
236 trace!("_ot_handle_active_scan_result: {:?}", result);
237
238 let result = ActiveScanResult::ref_from_ot_ptr(result);
240
241 let sender = &mut *(context as *mut F);
243
244 sender(result);
245 }
246
247 let (fn_ptr, fn_box, cb): (_, _, otHandleActiveScanResult) = {
248 let mut x = Box::new(f);
249
250 (
251 x.as_mut() as *mut F as *mut ::std::os::raw::c_void,
252 Some(x as Box<dyn FnMut(Option<&ActiveScanResult>) + 'a>),
253 Some(_ot_handle_active_scan_result::<F>),
254 )
255 };
256
257 unsafe {
258 Error::from(otLinkActiveScan(
259 self.as_ot_ptr(),
260 channels.into(),
261 dwell.as_millis().try_into().unwrap(),
262 cb,
263 fn_ptr,
264 ))
265 .into_result()?;
266
267 self.borrow_backing().active_scan_fn.set(std::mem::transmute::<
273 Option<Box<dyn FnMut(Option<&ActiveScanResult>) + 'a>>,
274 Option<Box<dyn FnMut(Option<&ActiveScanResult>) + 'static>>,
275 >(fn_box));
276 }
277
278 Ok(())
279 }
280
281 fn start_energy_scan<'a, F>(&self, channels: ChannelMask, dwell: Duration, f: F) -> Result
282 where
283 F: FnMut(Option<&EnergyScanResult>) + 'a,
284 {
285 unsafe extern "C" fn _ot_handle_energy_scan_result<
286 'a,
287 F: FnMut(Option<&EnergyScanResult>) + 'a,
288 >(
289 result: *mut otEnergyScanResult,
290 context: *mut ::std::os::raw::c_void,
291 ) {
292 trace!("_ot_handle_energy_scan_result: {:?}", result);
293
294 let result = EnergyScanResult::ref_from_ot_ptr(result);
296
297 let sender = &mut *(context as *mut F);
299
300 sender(result);
301 }
302
303 let (fn_ptr, fn_box, cb): (_, _, otHandleEnergyScanResult) = {
304 let mut x = Box::new(f);
305
306 (
307 x.as_mut() as *mut F as *mut ::std::os::raw::c_void,
308 Some(x as Box<dyn FnMut(Option<&EnergyScanResult>) + 'a>),
309 Some(_ot_handle_energy_scan_result::<F>),
310 )
311 };
312
313 unsafe {
314 Error::from(otLinkEnergyScan(
315 self.as_ot_ptr(),
316 channels.into(),
317 dwell.as_millis().try_into().unwrap(),
318 cb,
319 fn_ptr,
320 ))
321 .into_result()?;
322
323 self.borrow_backing().energy_scan_fn.set(std::mem::transmute::<
329 Option<Box<dyn FnMut(Option<&EnergyScanResult>) + 'a>>,
330 Option<Box<dyn FnMut(Option<&EnergyScanResult>) + 'static>>,
331 >(fn_box));
332 }
333
334 Ok(())
335 }
336}