1use crate::prelude_internal::*;
6
7use core::fmt::{Debug, Formatter};
8use num_derive::FromPrimitive;
9use std::net::SocketAddrV6;
10
11pub type Ip6Address = std::net::Ipv6Addr;
31
32pub const IP6_PREFIX_BITSIZE: u8 = otsys::OT_IP6_PREFIX_BITSIZE as u8;
34
35impl Transparent for std::net::Ipv6Addr {
36 fn from_ot(x: otIp6Address) -> std::net::Ipv6Addr {
37 unsafe { x.mFields.m8.into() }
38 }
39
40 fn into_ot(self) -> otIp6Address {
41 otIp6Address { mFields: otIp6Address__bindgen_ty_1 { m8: self.octets() } }
42 }
43}
44
45unsafe impl OtCastable for std::net::Ipv6Addr {
46 type OtType = otIp6Address;
47
48 fn as_ot_ptr(&self) -> *const otIp6Address {
49 sa::assert_eq_size!(Ip6Address, otIp6Address);
50 sa::assert_eq_align!(Ip6Address, otIp6Address);
51 self as *const Self as *const otIp6Address
52 }
53
54 fn as_ot_mut_ptr(&mut self) -> *mut Self::OtType {
55 self as *mut Self as *mut Self::OtType
56 }
57
58 unsafe fn ref_from_ot_ptr<'a>(ptr: *const otIp6Address) -> Option<&'a Self> {
59 if ptr.is_null() {
60 None
61 } else {
62 sa::assert_eq_size!(Ip6Address, otIp6Address);
63 sa::assert_eq_align!(Ip6Address, otIp6Address);
64
65 Some(&*(ptr as *const Self))
67 }
68 }
69
70 unsafe fn mut_from_ot_mut_ptr<'a>(ptr: *mut otIp6Address) -> Option<&'a mut Self> {
71 if ptr.is_null() {
72 None
73 } else {
74 sa::assert_eq_size!(Ip6Address, otIp6Address);
75 sa::assert_eq_align!(Ip6Address, otIp6Address);
76
77 Some(&mut *(ptr as *mut Self))
79 }
80 }
81}
82
83#[derive(Default, Clone, Copy)]
87#[repr(transparent)]
88pub struct Ip6NetworkPrefix(pub otIp6NetworkPrefix);
89
90impl_ot_castable!(Ip6NetworkPrefix, otIp6NetworkPrefix);
91
92impl PartialEq for Ip6NetworkPrefix {
93 fn eq(&self, other: &Self) -> bool {
94 self.0.m8 == other.0.m8
95 }
96}
97
98impl Eq for Ip6NetworkPrefix {}
99
100impl Debug for Ip6NetworkPrefix {
101 fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
102 std::net::Ipv6Addr::from(*self).fmt(f)?;
103 write!(f, "/64")
104 }
105}
106
107impl std::fmt::Display for Ip6NetworkPrefix {
108 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
109 std::fmt::Debug::fmt(self, f)
110 }
111}
112
113impl Ip6NetworkPrefix {
114 pub fn as_slice(&self) -> &[u8] {
116 &self.0.m8
117 }
118
119 pub fn octets(&self) -> [u8; 8] {
121 let mut octets = [0u8; 8];
122 octets.clone_from_slice(self.as_slice());
123 octets
124 }
125
126 pub fn contains(&self, addr: &std::net::Ipv6Addr) -> bool {
128 self.0.m8 == addr.octets()[0..8]
129 }
130}
131
132impl From<[u8; 8]> for Ip6NetworkPrefix {
133 fn from(m8: [u8; 8]) -> Self {
134 Self(otIp6NetworkPrefix { m8 })
135 }
136}
137
138impl From<Ip6NetworkPrefix> for std::net::Ipv6Addr {
139 fn from(prefix: Ip6NetworkPrefix) -> Self {
140 let mut octets = [0u8; 16];
141 octets[0..8].clone_from_slice(prefix.as_slice());
142 octets.into()
143 }
144}
145
146impl From<std::net::Ipv6Addr> for Ip6NetworkPrefix {
147 fn from(x: std::net::Ipv6Addr) -> Self {
148 let mut ret = Ip6NetworkPrefix::default();
149 ret.0.m8.clone_from_slice(&x.octets()[0..8]);
150 ret
151 }
152}
153
154#[derive(Default, Clone, Copy)]
160#[repr(transparent)]
161pub struct Ip6AddressInfo<'a>(otIp6AddressInfo, PhantomData<*mut &'a ()>);
162
163impl_ot_castable!(lifetime Ip6AddressInfo<'_>, otIp6AddressInfo, Default::default());
164
165impl Debug for Ip6AddressInfo<'_> {
166 fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
167 self.addr().fmt(f)?;
168 write!(f, "/{} {:?}", self.prefix_len(), self.scope())?;
169 if self.is_preferred() {
170 write!(f, " PREFERRED")?;
171 }
172
173 Ok(())
174 }
175}
176
177impl<'a> Ip6AddressInfo<'a> {
178 pub fn addr(&self) -> &'a Ip6Address {
180 unsafe { Ip6Address::ref_from_ot_ptr(self.0.mAddress).unwrap() }
181 }
182
183 pub fn is_preferred(&self) -> bool {
185 self.0.mPreferred()
186 }
187
188 pub fn prefix_len(&self) -> u8 {
190 self.0.mPrefixLength
191 }
192
193 pub fn scope(&self) -> Scope {
195 Scope(self.0.mScope())
196 }
197
198 pub fn is_multicast(&self) -> bool {
200 self.addr().is_multicast()
201 }
202}
203
204#[derive(Copy, Clone, Eq, PartialEq, Ord, PartialOrd)]
206#[repr(transparent)]
207pub struct Scope(pub u8);
208
209#[allow(missing_docs)]
210impl Scope {
211 pub const INTERFACE_LOCAL: Scope = Scope(0x1);
212 pub const LINK_LOCAL: Scope = Scope(0x2);
213 pub const REALM_LOCAL: Scope = Scope(0x3);
214 pub const ADMIN_LOCAL: Scope = Scope(0x4);
215 pub const SITE_LOCAL: Scope = Scope(0x5);
216 pub const ORGANIZATION_LOCAL: Scope = Scope(0x8);
217 pub const GLOBAL: Scope = Scope(0xe);
218}
219
220impl Debug for Scope {
221 fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
222 match *self {
223 Self::INTERFACE_LOCAL => write!(f, "Scope::INTERFACE_LOCAL"),
224 Self::LINK_LOCAL => write!(f, "Scope::LINK_LOCAL"),
225 Self::REALM_LOCAL => write!(f, "Scope::REALM_LOCAL"),
226 Self::ADMIN_LOCAL => write!(f, "Scope::ADMIN_LOCAL"),
227 Self::SITE_LOCAL => write!(f, "Scope::SITE_LOCAL"),
228 Self::ORGANIZATION_LOCAL => write!(f, "Scope::ORGANIZATION_LOCAL"),
229 Self::GLOBAL => write!(f, "Scope::GLOBAL"),
230 Scope(x) => write!(f, "Scope({x})"),
231 }
232 }
233}
234
235impl From<Scope> for u8 {
236 fn from(x: Scope) -> Self {
237 x.0
238 }
239}
240
241impl From<Scope> for u32 {
242 fn from(x: Scope) -> Self {
243 x.0 as u32
244 }
245}
246
247#[derive(Copy, Clone, Eq, PartialEq, Ord, PartialOrd)]
249pub struct AddressOrigin(pub u8);
250
251impl AddressOrigin {
252 pub const DHCPV6: AddressOrigin = AddressOrigin(OT_ADDRESS_ORIGIN_DHCPV6 as u8);
255
256 pub const MANUAL: AddressOrigin = AddressOrigin(OT_ADDRESS_ORIGIN_MANUAL as u8);
259
260 pub const SLAAC: AddressOrigin = AddressOrigin(OT_ADDRESS_ORIGIN_SLAAC as u8);
263
264 pub const THREAD: AddressOrigin = AddressOrigin(OT_ADDRESS_ORIGIN_THREAD as u8);
267}
268
269impl Debug for AddressOrigin {
270 fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
271 match *self {
272 Self::DHCPV6 => write!(f, "AddressOrigin::DHCPV6"),
273 Self::MANUAL => write!(f, "AddressOrigin::MANUAL"),
274 Self::SLAAC => write!(f, "AddressOrigin::SLAAC"),
275 Self::THREAD => write!(f, "AddressOrigin::THREAD"),
276 AddressOrigin(x) => write!(f, "AddressOrigin({x})"),
277 }
278 }
279}
280
281#[derive(Default, Clone, Copy)]
284#[repr(transparent)]
285pub struct NetifAddress(pub(crate) otNetifAddress);
286
287impl_ot_castable!(NetifAddress, otNetifAddress);
288
289impl Debug for NetifAddress {
290 fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
291 self.addr().fmt(f)?;
292 write!(f, "/{} {:?}", self.prefix_len(), self.address_origin())?;
293 if let Some(scope) = self.scope() {
294 write!(f, " {scope:?}")?;
295 }
296 if self.is_valid() {
297 write!(f, " VALID")?;
298 }
299 if self.is_preferred() {
300 write!(f, " PREFERRED")?;
301 }
302 if self.is_rloc() {
303 write!(f, " RLOC")?;
304 }
305 Ok(())
306 }
307}
308
309impl std::fmt::Display for NetifAddress {
310 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
311 std::fmt::Debug::fmt(self, f)
312 }
313}
314
315impl NetifAddress {
316 pub fn new(addr: std::net::Ipv6Addr, prefix_len: u8) -> NetifAddress {
320 let mut ret = NetifAddress(otNetifAddress {
321 mAddress: addr.into_ot(),
322 mPrefixLength: prefix_len,
323 mAddressOrigin: 0,
324 _bitfield_align_1: [],
325 _bitfield_1: Default::default(),
326 mNext: std::ptr::null_mut(),
327 });
328 ret.set_valid(true);
329 ret.set_preferred(true);
330 ret
331 }
332
333 pub fn addr(&self) -> &Ip6Address {
335 Ip6Address::ref_from_ot_ref(&self.0.mAddress)
336 }
337
338 pub fn prefix_len(&self) -> u8 {
340 self.0.mPrefixLength
341 }
342
343 pub fn address_origin(&self) -> AddressOrigin {
345 AddressOrigin(self.0.mAddressOrigin)
346 }
347
348 pub fn scope(&self) -> Option<Scope> {
350 if self.0.mScopeOverrideValid() {
351 Some(Scope(self.0.mScopeOverride().try_into().expect("NetifAddress: invalid scope")))
352 } else {
353 None
354 }
355 }
356
357 pub fn set_scope(&mut self, scope: Option<Scope>) {
359 if let Some(scope) = scope {
360 self.0.set_mScopeOverrideValid(true);
361 self.0.set_mScopeOverride(scope.into());
362 } else {
363 self.0.set_mScopeOverrideValid(false);
364 }
365 }
366
367 pub fn is_preferred(&self) -> bool {
369 self.0.mPreferred()
370 }
371
372 pub fn set_preferred(&mut self, x: bool) {
374 self.0.set_mPreferred(x)
375 }
376
377 pub fn is_rloc(&self) -> bool {
379 self.0.mRloc()
380 }
381
382 pub fn set_rloc(&mut self, x: bool) {
384 self.0.set_mRloc(x)
385 }
386
387 pub fn is_valid(&self) -> bool {
389 self.0.mValid()
390 }
391
392 pub fn set_valid(&mut self, x: bool) {
394 self.0.set_mValid(x)
395 }
396}
397
398#[derive(Default, Clone, Copy)]
401#[repr(transparent)]
402pub struct Ip6Prefix(pub otIp6Prefix);
403
404impl_ot_castable!(Ip6Prefix, otIp6Prefix);
405
406impl PartialEq for Ip6Prefix {
407 fn eq(&self, other: &Self) -> bool {
408 self.addr() == other.addr() && self.prefix_len() == other.prefix_len()
409 }
410}
411
412impl Eq for Ip6Prefix {}
413
414impl Debug for Ip6Prefix {
415 fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
416 self.addr().fmt(f)?;
417 write!(f, "/{}", self.prefix_len())
418 }
419}
420
421impl std::fmt::Display for Ip6Prefix {
422 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
423 std::fmt::Debug::fmt(self, f)
424 }
425}
426
427impl Ip6Prefix {
428 pub fn new<T: Into<Ip6Address>>(addr: T, prefix_len: u8) -> Ip6Prefix {
430 Ip6Prefix(otIp6Prefix { mPrefix: addr.into().into_ot(), mLength: prefix_len })
431 }
432
433 pub fn addr(&self) -> &Ip6Address {
435 Ip6Address::ref_from_ot_ref(&self.0.mPrefix)
436 }
437
438 pub fn prefix_len(&self) -> u8 {
440 self.0.mLength
441 }
442}
443
444#[derive(Default, Clone, Copy)]
446#[repr(transparent)]
447pub struct SockAddr(pub otSockAddr);
448
449impl_ot_castable!(SockAddr, otSockAddr);
450
451impl PartialEq for SockAddr {
452 fn eq(&self, other: &Self) -> bool {
453 self.addr() == other.addr() && self.port() == other.port()
454 }
455}
456
457impl Eq for SockAddr {}
458
459impl Debug for SockAddr {
460 fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
461 write!(f, "[")?;
462 self.addr().fmt(f)?;
463 write!(f, "]:{}", self.port())
464 }
465}
466
467impl std::fmt::Display for SockAddr {
468 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
469 std::fmt::Debug::fmt(self, f)
470 }
471}
472
473impl SockAddr {
474 pub fn new(addr: Ip6Address, port: u16) -> Self {
476 SockAddr(otSockAddr { mAddress: addr.into_ot(), mPort: port })
477 }
478
479 pub fn addr(&self) -> Ip6Address {
481 Ip6Address::from_ot(self.0.mAddress)
482 }
483
484 pub fn port(&self) -> u16 {
486 self.0.mPort
487 }
488}
489
490impl From<(Ip6Address, u16)> for SockAddr {
491 fn from(x: (Ip6Address, u16)) -> Self {
492 Self::new(x.0, x.1)
493 }
494}
495
496impl From<Ip6Address> for SockAddr {
497 fn from(x: Ip6Address) -> Self {
498 Self::new(x, 0)
499 }
500}
501
502impl From<std::net::SocketAddrV6> for SockAddr {
503 fn from(x: std::net::SocketAddrV6) -> Self {
504 SockAddr::new(*x.ip(), x.port())
505 }
506}
507
508impl From<SockAddr> for std::net::SocketAddrV6 {
509 fn from(x: SockAddr) -> Self {
510 SocketAddrV6::new(x.addr(), x.port(), 0, 0)
511 }
512}
513
514impl From<SockAddr> for std::net::SocketAddr {
515 fn from(x: SockAddr) -> Self {
516 std::net::SocketAddr::V6(x.into())
517 }
518}
519
520#[derive(Debug, Copy, Clone, Eq, Ord, PartialOrd, PartialEq, num_derive::FromPrimitive)]
524#[allow(missing_docs)]
525pub enum NetifIdentifier {
526 Backbone = OT_NETIF_BACKBONE as isize,
528
529 Thread = OT_NETIF_THREAD_HOST as isize,
531
532 Unspecified = OT_NETIF_UNSPECIFIED as isize,
534}
535
536impl From<otNetifIdentifier> for NetifIdentifier {
537 fn from(x: otNetifIdentifier) -> Self {
538 use num::FromPrimitive;
539 Self::from_u32(x).unwrap_or_else(|| panic!("Unknown otNetifIdentifier value: {x}"))
540 }
541}
542
543impl From<NetifIdentifier> for otNetifIdentifier {
544 fn from(x: NetifIdentifier) -> Self {
545 x as otNetifIdentifier
546 }
547}
548
549#[derive(Default, Clone, Copy)]
552#[repr(transparent)]
553pub struct BorderRoutingPrefixTableEntry(pub otBorderRoutingPrefixTableEntry);
554
555impl_ot_castable!(BorderRoutingPrefixTableEntry, otBorderRoutingPrefixTableEntry);
556
557impl Debug for BorderRoutingPrefixTableEntry {
558 fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
559 f.debug_struct("otBorderRoutingPrefixTableEntry")
560 .field("router_address", &self.router_address())
561 .field("prefix", &self.prefix())
562 .field("is_on_link", &self.is_on_link())
563 .field("msec_since_last_update", &self.msec_since_last_update())
564 .field("valid_lifetime", &self.valid_lifetime())
565 .field("preferred_lifetime", &self.preferred_lifetime())
566 .field("route_preference", &self.route_preference())
567 .finish()
568 }
569}
570
571impl std::fmt::Display for BorderRoutingPrefixTableEntry {
572 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
573 std::fmt::Debug::fmt(self, f)
574 }
575}
576
577impl BorderRoutingPrefixTableEntry {
578 pub fn router_address(&self) -> &Ip6Address {
580 Ip6Address::ref_from_ot_ref(&self.0.mRouter.mAddress)
581 }
582
583 pub fn prefix(&self) -> &Ip6Prefix {
585 Ip6Prefix::ref_from_ot_ref(&self.0.mPrefix)
586 }
587
588 pub fn is_on_link(&self) -> bool {
590 self.0.mIsOnLink
591 }
592
593 pub fn msec_since_last_update(&self) -> u32 {
595 self.0.mMsecSinceLastUpdate
596 }
597
598 pub fn valid_lifetime(&self) -> u32 {
600 self.0.mValidLifetime
601 }
602
603 pub fn preferred_lifetime(&self) -> u32 {
605 self.0.mPreferredLifetime
606 }
607
608 pub fn route_preference(&self) -> RoutePreference {
610 RoutePreference::from_isize(self.0.mPreferredLifetime as isize)
611 .unwrap_or(RoutePreference::Medium)
612 }
613}
614
615#[derive(Debug, Default, Copy, Clone, Eq, Ord, PartialOrd, PartialEq, FromPrimitive)]
619#[allow(missing_docs)]
620pub enum BorderRoutingDhcp6PdState {
621 #[default]
622 Disabled = OT_BORDER_ROUTING_DHCP6_PD_STATE_DISABLED as isize,
624
625 Stopped = OT_BORDER_ROUTING_DHCP6_PD_STATE_STOPPED as isize,
627
628 Running = OT_BORDER_ROUTING_DHCP6_PD_STATE_RUNNING as isize,
630}
631
632impl From<otBorderRoutingDhcp6PdState> for BorderRoutingDhcp6PdState {
633 fn from(x: otBorderRoutingDhcp6PdState) -> Self {
634 use num::FromPrimitive;
635 Self::from_u32(x)
636 .unwrap_or_else(|| panic!("Unknown otBorderRoutingDhcp6PdState value: {x}"))
637 }
638}
639
640impl From<BorderRoutingDhcp6PdState> for otBorderRoutingDhcp6PdState {
641 fn from(x: BorderRoutingDhcp6PdState) -> Self {
642 x as otBorderRoutingDhcp6PdState
643 }
644}