1use crate::prelude_internal::*;
6
7#[derive(Debug, Copy, Clone, Eq, Ord, PartialOrd, PartialEq, num_derive::FromPrimitive)]
9pub enum Icmp6EchoMode {
10 HandleAll = OT_ICMP6_ECHO_HANDLER_ALL as isize,
12
13 HandleDisabled = OT_ICMP6_ECHO_HANDLER_DISABLED as isize,
15
16 HandleMulticastOnly = OT_ICMP6_ECHO_HANDLER_MULTICAST_ONLY as isize,
18
19 HandleUnicastOnly = OT_ICMP6_ECHO_HANDLER_UNICAST_ONLY as isize,
21}
22
23impl From<otIcmp6EchoMode> for Icmp6EchoMode {
24 fn from(x: otIcmp6EchoMode) -> Self {
25 use num::FromPrimitive;
26 Self::from_u32(x).unwrap_or_else(|| panic!("Unknown otIcmp6EchoMode value: {x}"))
27 }
28}
29
30impl From<Icmp6EchoMode> for otIcmp6EchoMode {
31 fn from(x: Icmp6EchoMode) -> Self {
32 x as otIcmp6EchoMode
33 }
34}
35
36#[derive(Default, Debug, Clone)]
38pub struct NetifAddressIterator<'a>(Option<&'a NetifAddress>);
39
40impl<'a> Iterator for NetifAddressIterator<'a> {
41 type Item = &'a NetifAddress;
42
43 fn next(&mut self) -> Option<Self::Item> {
44 if let Some(addr_ref) = self.0 {
45 self.0 = unsafe { NetifAddress::ref_from_ot_ptr(addr_ref.0.mNext) };
49 Some(addr_ref)
50 } else {
51 None
52 }
53 }
54}
55
56pub trait Ip6 {
58 fn ip6_send(&self, message: OtMessageBox<'_>) -> Result;
60
61 fn ip6_send_data(&self, data: &[u8]) -> Result;
64
65 fn ip6_send_data_direct(&self, data: &[u8]) -> Result;
67
68 fn ip6_is_enabled(&self) -> bool;
70
71 fn ip6_set_enabled(&self, enabled: bool) -> Result;
73
74 fn ip6_add_unicast_address(&self, addr: &NetifAddress) -> Result;
77
78 fn ip6_remove_unicast_address(&self, addr: &Ip6Address) -> Result;
81
82 fn ip6_join_multicast_group(&self, addr: &Ip6Address) -> Result;
85
86 fn ip6_leave_multicast_group(&self, addr: &Ip6Address) -> Result;
89
90 fn ip6_set_receive_fn<'a, F>(&'a self, f: Option<F>)
96 where
97 F: FnMut(OtMessageBox<'_>) + 'a;
98
99 fn ip6_set_address_fn<'a, F>(&'a self, f: Option<F>)
110 where
111 F: for<'r> FnMut(Ip6AddressInfo<'r>, bool) + 'a;
112
113 fn ip6_is_slaac_enabled(&self) -> bool;
115
116 fn ip6_set_slaac_enabled(&self, enabled: bool);
119
120 fn icmp6_get_echo_mode(&self) -> Icmp6EchoMode;
123
124 fn icmp6_set_echo_mode(&self, mode: Icmp6EchoMode);
127
128 fn ip6_set_receive_filter_enabled(&self, enabled: bool);
131
132 fn ip6_is_receive_filter_enabled(&self) -> bool;
135
136 fn ip6_get_border_routing_counters(&self) -> &BorderRoutingCounters;
139
140 fn ip6_get_unicast_addresses(&self) -> NetifAddressIterator<'_>;
143}
144
145impl<T: Ip6 + ot::Boxable> Ip6 for ot::Box<T> {
146 fn ip6_send(&self, message: OtMessageBox<'_>) -> Result<()> {
147 self.as_ref().ip6_send(message)
148 }
149
150 fn ip6_send_data(&self, data: &[u8]) -> Result {
151 self.as_ref().ip6_send_data(data)
152 }
153
154 fn ip6_send_data_direct(&self, data: &[u8]) -> Result {
155 self.as_ref().ip6_send_data_direct(data)
156 }
157
158 fn ip6_is_enabled(&self) -> bool {
159 self.as_ref().ip6_is_enabled()
160 }
161
162 fn ip6_set_enabled(&self, enabled: bool) -> Result {
163 self.as_ref().ip6_set_enabled(enabled)
164 }
165
166 fn ip6_add_unicast_address(&self, addr: &NetifAddress) -> Result {
167 self.as_ref().ip6_add_unicast_address(addr)
168 }
169
170 fn ip6_remove_unicast_address(&self, addr: &std::net::Ipv6Addr) -> Result {
171 self.as_ref().ip6_remove_unicast_address(addr)
172 }
173
174 fn ip6_join_multicast_group(&self, addr: &std::net::Ipv6Addr) -> Result {
175 self.as_ref().ip6_join_multicast_group(addr)
176 }
177
178 fn ip6_leave_multicast_group(&self, addr: &std::net::Ipv6Addr) -> Result {
179 self.as_ref().ip6_join_multicast_group(addr)
180 }
181
182 fn ip6_set_receive_fn<'a, F>(&'a self, f: Option<F>)
183 where
184 F: FnMut(OtMessageBox<'_>) + 'a,
185 {
186 self.as_ref().ip6_set_receive_fn(f)
187 }
188
189 fn ip6_set_address_fn<'a, F>(&'a self, f: Option<F>)
190 where
191 F: for<'r> FnMut(Ip6AddressInfo<'r>, bool) + 'a,
192 {
193 self.as_ref().ip6_set_address_fn(f)
194 }
195
196 fn ip6_is_slaac_enabled(&self) -> bool {
197 self.as_ref().ip6_is_slaac_enabled()
198 }
199
200 fn ip6_set_slaac_enabled(&self, enabled: bool) {
201 self.as_ref().ip6_set_slaac_enabled(enabled);
202 }
203
204 fn icmp6_get_echo_mode(&self) -> Icmp6EchoMode {
205 self.as_ref().icmp6_get_echo_mode()
206 }
207
208 fn icmp6_set_echo_mode(&self, mode: Icmp6EchoMode) {
209 self.as_ref().icmp6_set_echo_mode(mode)
210 }
211
212 fn ip6_set_receive_filter_enabled(&self, enabled: bool) {
213 self.as_ref().ip6_set_receive_filter_enabled(enabled);
214 }
215
216 fn ip6_is_receive_filter_enabled(&self) -> bool {
217 self.as_ref().ip6_is_receive_filter_enabled()
218 }
219
220 fn ip6_get_border_routing_counters(&self) -> &BorderRoutingCounters {
221 self.as_ref().ip6_get_border_routing_counters()
222 }
223
224 fn ip6_get_unicast_addresses(&self) -> NetifAddressIterator<'_> {
225 self.as_ref().ip6_get_unicast_addresses()
226 }
227}
228
229impl Ip6 for Instance {
230 fn ip6_send(&self, message: OtMessageBox<'_>) -> Result {
231 Error::from(unsafe { otIp6Send(self.as_ot_ptr(), message.take_ot_ptr()) }).into()
232 }
233
234 fn ip6_send_data(&self, data: &[u8]) -> Result {
235 if let Ok(msg) = Message::ip6_new_from_bytes(self, data, None) {
236 self.ip6_send(msg)
237 } else if self.get_buffer_info().0.mFreeBuffers == 0 {
238 Err(ot::Error::NoBufs)
239 } else {
240 Err(ot::Error::Failed)
241 }
242 }
243
244 fn ip6_send_data_direct(&self, data: &[u8]) -> Result {
245 if let Ok(mut msg) = Message::ip6_new_from_bytes(self, data, None) {
246 msg.set_direct_transmission(true);
247 self.ip6_send(msg)
248 } else if self.get_buffer_info().0.mFreeBuffers == 0 {
249 Err(ot::Error::NoBufs)
250 } else {
251 Err(ot::Error::Failed)
252 }
253 }
254
255 fn ip6_is_enabled(&self) -> bool {
256 unsafe { otIp6IsEnabled(self.as_ot_ptr()) }
257 }
258
259 fn ip6_set_enabled(&self, enabled: bool) -> Result {
260 Error::from(unsafe { otIp6SetEnabled(self.as_ot_ptr(), enabled) }).into()
261 }
262
263 fn ip6_add_unicast_address(&self, addr: &NetifAddress) -> Result {
264 Error::from(unsafe { otIp6AddUnicastAddress(self.as_ot_ptr(), addr.as_ot_ptr()) }).into()
265 }
266
267 fn ip6_remove_unicast_address(&self, addr: &std::net::Ipv6Addr) -> Result {
268 Error::from(unsafe { otIp6RemoveUnicastAddress(self.as_ot_ptr(), addr.as_ot_ptr()) }).into()
269 }
270
271 fn ip6_join_multicast_group(&self, addr: &std::net::Ipv6Addr) -> Result {
272 Error::from(unsafe { otIp6SubscribeMulticastAddress(self.as_ot_ptr(), addr.as_ot_ptr()) })
273 .into()
274 }
275
276 fn ip6_leave_multicast_group(&self, addr: &std::net::Ipv6Addr) -> Result {
277 Error::from(unsafe { otIp6UnsubscribeMulticastAddress(self.as_ot_ptr(), addr.as_ot_ptr()) })
278 .into()
279 }
280
281 fn ip6_set_receive_fn<'a, F>(&'a self, f: Option<F>)
282 where
283 F: FnMut(OtMessageBox<'_>) + 'a,
284 {
285 unsafe extern "C" fn _ot_ip6_receive_callback<'a, F: FnMut(OtMessageBox<'_>) + 'a>(
286 message: *mut otMessage,
287 context: *mut ::std::os::raw::c_void,
288 ) {
289 trace!("_ot_ip6_receive_callback");
290
291 let message = OtMessageBox::from_ot_ptr(message)
293 .expect("_ot_ip6_receive_callback: Got NULL otMessage");
294
295 let sender = &mut *(context as *mut F);
297
298 sender(message)
299 }
300
301 let (fn_ptr, fn_box, cb): (_, _, otIp6ReceiveCallback) = if let Some(f) = f {
302 let mut x = Box::new(f);
303
304 (
305 x.as_mut() as *mut F as *mut ::std::os::raw::c_void,
306 Some(x as Box<dyn FnMut(OtMessageBox<'_>) + 'a>),
307 Some(_ot_ip6_receive_callback::<F>),
308 )
309 } else {
310 (std::ptr::null_mut() as *mut ::std::os::raw::c_void, None, None)
311 };
312
313 unsafe {
314 otIp6SetReceiveCallback(self.as_ot_ptr(), cb, fn_ptr);
315
316 self.borrow_backing().ip6_receive_fn.set(std::mem::transmute::<
322 Option<Box<dyn FnMut(OtMessageBox<'_>) + 'a>>,
323 Option<Box<dyn FnMut(OtMessageBox<'_>) + 'static>>,
324 >(fn_box));
325 }
326 }
327
328 fn ip6_set_address_fn<'a, F>(&'a self, f: Option<F>)
329 where
330 F: for<'r> FnMut(Ip6AddressInfo<'r>, bool) + 'a,
331 {
332 unsafe extern "C" fn _ot_ip6_address_callback<
333 'a,
334 F: FnMut(Ip6AddressInfo<'_>, bool) + 'a,
335 >(
336 info: *const otIp6AddressInfo,
337 is_added: bool,
338 context: *mut ::std::os::raw::c_void,
339 ) {
340 trace!("_ot_ip6_address_callback");
341
342 let info = *Ip6AddressInfo::ref_from_ot_ptr(info).unwrap();
344
345 let sender = &mut *(context as *mut F);
347
348 sender(info, is_added)
349 }
350
351 let (fn_ptr, fn_box, cb): (_, _, otIp6AddressCallback) = if let Some(f) = f {
352 let mut x = Box::new(f);
353
354 (
355 x.as_mut() as *mut F as *mut ::std::os::raw::c_void,
356 Some(x as Box<dyn FnMut(Ip6AddressInfo<'_>, bool) + 'a>),
357 Some(_ot_ip6_address_callback::<F>),
358 )
359 } else {
360 (std::ptr::null_mut() as *mut ::std::os::raw::c_void, None, None)
361 };
362
363 unsafe {
364 otIp6SetAddressCallback(self.as_ot_ptr(), cb, fn_ptr);
365
366 self.borrow_backing().ip6_address_fn.set(std::mem::transmute::<
372 Option<Box<dyn FnMut(Ip6AddressInfo<'_>, bool) + 'a>>,
373 Option<Box<dyn FnMut(Ip6AddressInfo<'_>, bool) + 'static>>,
374 >(fn_box));
375 }
376 }
377
378 fn ip6_is_slaac_enabled(&self) -> bool {
379 unsafe { otIp6IsSlaacEnabled(self.as_ot_ptr()) }
380 }
381
382 fn ip6_set_slaac_enabled(&self, enabled: bool) {
383 unsafe { otIp6SetSlaacEnabled(self.as_ot_ptr(), enabled) }
384 }
385
386 fn icmp6_get_echo_mode(&self) -> Icmp6EchoMode {
387 unsafe { otIcmp6GetEchoMode(self.as_ot_ptr()) }.into()
388 }
389
390 fn icmp6_set_echo_mode(&self, mode: Icmp6EchoMode) {
391 unsafe { otIcmp6SetEchoMode(self.as_ot_ptr(), mode.into()) }
392 }
393
394 fn ip6_set_receive_filter_enabled(&self, enabled: bool) {
395 unsafe { otIp6SetReceiveFilterEnabled(self.as_ot_ptr(), enabled) }
396 }
397
398 fn ip6_is_receive_filter_enabled(&self) -> bool {
399 unsafe { otIp6IsReceiveFilterEnabled(self.as_ot_ptr()) }
400 }
401
402 fn ip6_get_border_routing_counters(&self) -> &BorderRoutingCounters {
403 unsafe {
404 BorderRoutingCounters::ref_from_ot_ptr(otIp6GetBorderRoutingCounters(self.as_ot_ptr()))
405 .unwrap()
406 }
407 }
408
409 fn ip6_get_unicast_addresses(&self) -> NetifAddressIterator<'_> {
410 NetifAddressIterator(unsafe {
411 NetifAddress::ref_from_ot_ptr(otIp6GetUnicastAddresses(self.as_ot_ptr()))
412 })
413 }
414}