mod.rs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371
  1. //! Implementation of [RFC 6282] which specifies a compression format for IPv6 datagrams over
  2. //! IEEE802.154-based networks.
  3. //!
  4. //! [RFC 6282]: https://datatracker.ietf.org/doc/html/rfc6282
  5. use super::{Error, Result};
  6. use crate::wire::ieee802154::Address as LlAddress;
  7. use crate::wire::ipv6;
  8. use crate::wire::IpProtocol;
  9. pub mod frag;
  10. pub mod iphc;
  11. pub mod nhc;
  12. const ADDRESS_CONTEXT_LENGTH: usize = 8;
  13. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  14. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  15. pub struct AddressContext(pub [u8; ADDRESS_CONTEXT_LENGTH]);
  16. /// The representation of an unresolved address. 6LoWPAN compression of IPv6 addresses can be with
  17. /// and without context information. The decompression with context information is not yet
  18. /// implemented.
  19. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  20. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  21. pub enum UnresolvedAddress<'a> {
  22. WithoutContext(AddressMode<'a>),
  23. WithContext((usize, AddressMode<'a>)),
  24. Reserved,
  25. }
  26. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  27. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  28. pub enum AddressMode<'a> {
  29. /// The full address is carried in-line.
  30. FullInline(&'a [u8]),
  31. /// The first 64-bits of the address are elided. The value of those bits
  32. /// is the link-local prefix padded with zeros. The remaining 64 bits are
  33. /// carried in-line.
  34. InLine64bits(&'a [u8]),
  35. /// The first 112 bits of the address are elided. The value of the first
  36. /// 64 bits is the link-local prefix padded with zeros. The following 64 bits
  37. /// are 0000:00ff:fe00:XXXX, where XXXX are the 16 bits carried in-line.
  38. InLine16bits(&'a [u8]),
  39. /// The address is fully elided. The first 64 bits of the address are
  40. /// the link-local prefix padded with zeros. The remaining 64 bits are
  41. /// computed from the encapsulating header (e.g., 802.15.4 or IPv6 source address)
  42. /// as specified in Section 3.2.2.
  43. FullyElided,
  44. /// The address takes the form ffXX::00XX:XXXX:XXXX
  45. Multicast48bits(&'a [u8]),
  46. /// The address takes the form ffXX::00XX:XXXX.
  47. Multicast32bits(&'a [u8]),
  48. /// The address takes the form ff02::00XX.
  49. Multicast8bits(&'a [u8]),
  50. /// The unspecified address.
  51. Unspecified,
  52. NotSupported,
  53. }
  54. const LINK_LOCAL_PREFIX: [u8; 2] = [0xfe, 0x80];
  55. const EUI64_MIDDLE_VALUE: [u8; 2] = [0xff, 0xfe];
  56. impl<'a> UnresolvedAddress<'a> {
  57. pub fn resolve(
  58. self,
  59. ll_address: Option<LlAddress>,
  60. addr_context: &[AddressContext],
  61. ) -> Result<ipv6::Address> {
  62. let mut bytes = [0; 16];
  63. let copy_context = |index: usize, bytes: &mut [u8]| -> Result<()> {
  64. if index >= addr_context.len() {
  65. return Err(Error);
  66. }
  67. let context = addr_context[index];
  68. bytes[..ADDRESS_CONTEXT_LENGTH].copy_from_slice(&context.0);
  69. Ok(())
  70. };
  71. match self {
  72. UnresolvedAddress::WithoutContext(mode) => match mode {
  73. AddressMode::FullInline(addr) => Ok(ipv6::Address::from_bytes(addr)),
  74. AddressMode::InLine64bits(inline) => {
  75. bytes[0..2].copy_from_slice(&LINK_LOCAL_PREFIX[..]);
  76. bytes[8..].copy_from_slice(inline);
  77. Ok(ipv6::Address::from_bytes(&bytes[..]))
  78. }
  79. AddressMode::InLine16bits(inline) => {
  80. bytes[0..2].copy_from_slice(&LINK_LOCAL_PREFIX[..]);
  81. bytes[11..13].copy_from_slice(&EUI64_MIDDLE_VALUE[..]);
  82. bytes[14..].copy_from_slice(inline);
  83. Ok(ipv6::Address::from_bytes(&bytes[..]))
  84. }
  85. AddressMode::FullyElided => {
  86. bytes[0..2].copy_from_slice(&LINK_LOCAL_PREFIX[..]);
  87. match ll_address {
  88. Some(LlAddress::Short(ll)) => {
  89. bytes[11..13].copy_from_slice(&EUI64_MIDDLE_VALUE[..]);
  90. bytes[14..].copy_from_slice(&ll);
  91. }
  92. Some(addr @ LlAddress::Extended(_)) => match addr.as_eui_64() {
  93. Some(addr) => bytes[8..].copy_from_slice(&addr),
  94. None => return Err(Error),
  95. },
  96. Some(LlAddress::Absent) => return Err(Error),
  97. None => return Err(Error),
  98. }
  99. Ok(ipv6::Address::from_bytes(&bytes[..]))
  100. }
  101. AddressMode::Multicast48bits(inline) => {
  102. bytes[0] = 0xff;
  103. bytes[1] = inline[0];
  104. bytes[11..].copy_from_slice(&inline[1..][..5]);
  105. Ok(ipv6::Address::from_bytes(&bytes[..]))
  106. }
  107. AddressMode::Multicast32bits(inline) => {
  108. bytes[0] = 0xff;
  109. bytes[1] = inline[0];
  110. bytes[13..].copy_from_slice(&inline[1..][..3]);
  111. Ok(ipv6::Address::from_bytes(&bytes[..]))
  112. }
  113. AddressMode::Multicast8bits(inline) => {
  114. bytes[0] = 0xff;
  115. bytes[1] = 0x02;
  116. bytes[15] = inline[0];
  117. Ok(ipv6::Address::from_bytes(&bytes[..]))
  118. }
  119. _ => Err(Error),
  120. },
  121. UnresolvedAddress::WithContext(mode) => match mode {
  122. (_, AddressMode::Unspecified) => Ok(ipv6::Address::UNSPECIFIED),
  123. (index, AddressMode::InLine64bits(inline)) => {
  124. copy_context(index, &mut bytes[..])?;
  125. bytes[16 - inline.len()..].copy_from_slice(inline);
  126. Ok(ipv6::Address::from_bytes(&bytes[..]))
  127. }
  128. (index, AddressMode::InLine16bits(inline)) => {
  129. copy_context(index, &mut bytes[..])?;
  130. bytes[16 - inline.len()..].copy_from_slice(inline);
  131. Ok(ipv6::Address::from_bytes(&bytes[..]))
  132. }
  133. (index, AddressMode::FullyElided) => {
  134. match ll_address {
  135. Some(LlAddress::Short(ll)) => {
  136. bytes[11..13].copy_from_slice(&EUI64_MIDDLE_VALUE[..]);
  137. bytes[14..].copy_from_slice(&ll);
  138. }
  139. Some(addr @ LlAddress::Extended(_)) => match addr.as_eui_64() {
  140. Some(addr) => bytes[8..].copy_from_slice(&addr),
  141. None => return Err(Error),
  142. },
  143. Some(LlAddress::Absent) => return Err(Error),
  144. None => return Err(Error),
  145. }
  146. copy_context(index, &mut bytes[..])?;
  147. Ok(ipv6::Address::from_bytes(&bytes[..]))
  148. }
  149. _ => Err(Error),
  150. },
  151. UnresolvedAddress::Reserved => Err(Error),
  152. }
  153. }
  154. }
  155. #[derive(Debug, Clone, Copy, PartialEq, Eq)]
  156. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  157. pub enum SixlowpanPacket {
  158. FragmentHeader,
  159. IphcHeader,
  160. }
  161. const DISPATCH_FIRST_FRAGMENT_HEADER: u8 = 0b11000;
  162. const DISPATCH_FRAGMENT_HEADER: u8 = 0b11100;
  163. const DISPATCH_IPHC_HEADER: u8 = 0b011;
  164. const DISPATCH_UDP_HEADER: u8 = 0b11110;
  165. const DISPATCH_EXT_HEADER: u8 = 0b1110;
  166. impl SixlowpanPacket {
  167. /// Returns the type of the 6LoWPAN header.
  168. /// This can either be a fragment header or an IPHC header.
  169. ///
  170. /// # Errors
  171. /// Returns `[Error::Unrecognized]` when neither the Fragment Header dispatch or the IPHC
  172. /// dispatch is recognized.
  173. pub fn dispatch(buffer: impl AsRef<[u8]>) -> Result<Self> {
  174. let raw = buffer.as_ref();
  175. if raw.is_empty() {
  176. return Err(Error);
  177. }
  178. if raw[0] >> 3 == DISPATCH_FIRST_FRAGMENT_HEADER || raw[0] >> 3 == DISPATCH_FRAGMENT_HEADER
  179. {
  180. Ok(Self::FragmentHeader)
  181. } else if raw[0] >> 5 == DISPATCH_IPHC_HEADER {
  182. Ok(Self::IphcHeader)
  183. } else {
  184. Err(Error)
  185. }
  186. }
  187. }
  188. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  189. pub enum NextHeader {
  190. Compressed,
  191. Uncompressed(IpProtocol),
  192. }
  193. impl core::fmt::Display for NextHeader {
  194. fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
  195. match self {
  196. NextHeader::Compressed => write!(f, "compressed"),
  197. NextHeader::Uncompressed(protocol) => write!(f, "{protocol}"),
  198. }
  199. }
  200. }
  201. #[cfg(feature = "defmt")]
  202. impl defmt::Format for NextHeader {
  203. fn format(&self, fmt: defmt::Formatter) {
  204. match self {
  205. NextHeader::Compressed => defmt::write!(fmt, "compressed"),
  206. NextHeader::Uncompressed(protocol) => defmt::write!(fmt, "{}", protocol),
  207. }
  208. }
  209. }
  210. #[cfg(test)]
  211. mod test {
  212. use super::*;
  213. #[test]
  214. fn sixlowpan_fragment_emit() {
  215. let repr = frag::Repr::FirstFragment {
  216. size: 0xff,
  217. tag: 0xabcd,
  218. };
  219. let buffer = [0u8; 4];
  220. let mut packet = frag::Packet::new_unchecked(buffer);
  221. assert_eq!(repr.buffer_len(), 4);
  222. repr.emit(&mut packet);
  223. assert_eq!(packet.datagram_size(), 0xff);
  224. assert_eq!(packet.datagram_tag(), 0xabcd);
  225. assert_eq!(packet.into_inner(), [0xc0, 0xff, 0xab, 0xcd]);
  226. let repr = frag::Repr::Fragment {
  227. size: 0xff,
  228. tag: 0xabcd,
  229. offset: 0xcc,
  230. };
  231. let buffer = [0u8; 5];
  232. let mut packet = frag::Packet::new_unchecked(buffer);
  233. assert_eq!(repr.buffer_len(), 5);
  234. repr.emit(&mut packet);
  235. assert_eq!(packet.datagram_size(), 0xff);
  236. assert_eq!(packet.datagram_tag(), 0xabcd);
  237. assert_eq!(packet.into_inner(), [0xe0, 0xff, 0xab, 0xcd, 0xcc]);
  238. }
  239. #[test]
  240. fn sixlowpan_three_fragments() {
  241. use crate::wire::ieee802154::Frame as Ieee802154Frame;
  242. use crate::wire::ieee802154::Repr as Ieee802154Repr;
  243. use crate::wire::Ieee802154Address;
  244. let key = frag::Key {
  245. ll_src_addr: Ieee802154Address::Extended([50, 147, 130, 47, 40, 8, 62, 217]),
  246. ll_dst_addr: Ieee802154Address::Extended([26, 11, 66, 66, 66, 66, 66, 66]),
  247. datagram_size: 307,
  248. datagram_tag: 63,
  249. };
  250. let frame1: &[u8] = &[
  251. 0x41, 0xcc, 0x92, 0xef, 0xbe, 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x0b, 0x1a, 0xd9,
  252. 0x3e, 0x08, 0x28, 0x2f, 0x82, 0x93, 0x32, 0xc1, 0x33, 0x00, 0x3f, 0x6e, 0x33, 0x02,
  253. 0x35, 0x3d, 0xf0, 0xd2, 0x5f, 0x1b, 0x39, 0xb4, 0x6b, 0x4c, 0x6f, 0x72, 0x65, 0x6d,
  254. 0x20, 0x69, 0x70, 0x73, 0x75, 0x6d, 0x20, 0x64, 0x6f, 0x6c, 0x6f, 0x72, 0x20, 0x73,
  255. 0x69, 0x74, 0x20, 0x61, 0x6d, 0x65, 0x74, 0x2c, 0x20, 0x63, 0x6f, 0x6e, 0x73, 0x65,
  256. 0x63, 0x74, 0x65, 0x74, 0x75, 0x72, 0x20, 0x61, 0x64, 0x69, 0x70, 0x69, 0x73, 0x63,
  257. 0x69, 0x6e, 0x67, 0x20, 0x65, 0x6c, 0x69, 0x74, 0x2e, 0x20, 0x41, 0x6c, 0x69, 0x71,
  258. 0x75, 0x61, 0x6d, 0x20, 0x64, 0x75, 0x69, 0x20, 0x6f, 0x64, 0x69, 0x6f, 0x2c, 0x20,
  259. 0x69, 0x61, 0x63, 0x75, 0x6c, 0x69, 0x73, 0x20, 0x76, 0x65, 0x6c, 0x20, 0x72,
  260. ];
  261. let ieee802154_frame = Ieee802154Frame::new_checked(frame1).unwrap();
  262. let ieee802154_repr = Ieee802154Repr::parse(&ieee802154_frame).unwrap();
  263. let sixlowpan_frame =
  264. SixlowpanPacket::dispatch(ieee802154_frame.payload().unwrap()).unwrap();
  265. let frag = if let SixlowpanPacket::FragmentHeader = sixlowpan_frame {
  266. frag::Packet::new_checked(ieee802154_frame.payload().unwrap()).unwrap()
  267. } else {
  268. unreachable!()
  269. };
  270. assert_eq!(frag.datagram_size(), 307);
  271. assert_eq!(frag.datagram_tag(), 0x003f);
  272. assert_eq!(frag.datagram_offset(), 0);
  273. assert_eq!(frag.get_key(&ieee802154_repr), key);
  274. let frame2: &[u8] = &[
  275. 0x41, 0xcc, 0x93, 0xef, 0xbe, 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x0b, 0x1a, 0xd9,
  276. 0x3e, 0x08, 0x28, 0x2f, 0x82, 0x93, 0x32, 0xe1, 0x33, 0x00, 0x3f, 0x11, 0x75, 0x74,
  277. 0x72, 0x75, 0x6d, 0x20, 0x61, 0x74, 0x2c, 0x20, 0x74, 0x72, 0x69, 0x73, 0x74, 0x69,
  278. 0x71, 0x75, 0x65, 0x20, 0x6e, 0x6f, 0x6e, 0x20, 0x6e, 0x75, 0x6e, 0x63, 0x20, 0x65,
  279. 0x72, 0x61, 0x74, 0x20, 0x63, 0x75, 0x72, 0x61, 0x65, 0x2e, 0x20, 0x4c, 0x6f, 0x72,
  280. 0x65, 0x6d, 0x20, 0x69, 0x70, 0x73, 0x75, 0x6d, 0x20, 0x64, 0x6f, 0x6c, 0x6f, 0x72,
  281. 0x20, 0x73, 0x69, 0x74, 0x20, 0x61, 0x6d, 0x65, 0x74, 0x2c, 0x20, 0x63, 0x6f, 0x6e,
  282. 0x73, 0x65, 0x63, 0x74, 0x65, 0x74, 0x75, 0x72, 0x20, 0x61, 0x64, 0x69, 0x70, 0x69,
  283. 0x73, 0x63, 0x69, 0x6e, 0x67, 0x20, 0x65, 0x6c, 0x69, 0x74,
  284. ];
  285. let ieee802154_frame = Ieee802154Frame::new_checked(frame2).unwrap();
  286. let ieee802154_repr = Ieee802154Repr::parse(&ieee802154_frame).unwrap();
  287. let sixlowpan_frame =
  288. SixlowpanPacket::dispatch(ieee802154_frame.payload().unwrap()).unwrap();
  289. let frag = if let SixlowpanPacket::FragmentHeader = sixlowpan_frame {
  290. frag::Packet::new_checked(ieee802154_frame.payload().unwrap()).unwrap()
  291. } else {
  292. unreachable!()
  293. };
  294. assert_eq!(frag.datagram_size(), 307);
  295. assert_eq!(frag.datagram_tag(), 0x003f);
  296. assert_eq!(frag.datagram_offset(), 136 / 8);
  297. assert_eq!(frag.get_key(&ieee802154_repr), key);
  298. let frame3: &[u8] = &[
  299. 0x41, 0xcc, 0x94, 0xef, 0xbe, 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x0b, 0x1a, 0xd9,
  300. 0x3e, 0x08, 0x28, 0x2f, 0x82, 0x93, 0x32, 0xe1, 0x33, 0x00, 0x3f, 0x1d, 0x2e, 0x20,
  301. 0x41, 0x6c, 0x69, 0x71, 0x75, 0x61, 0x6d, 0x20, 0x64, 0x75, 0x69, 0x20, 0x6f, 0x64,
  302. 0x69, 0x6f, 0x2c, 0x20, 0x69, 0x61, 0x63, 0x75, 0x6c, 0x69, 0x73, 0x20, 0x76, 0x65,
  303. 0x6c, 0x20, 0x72, 0x75, 0x74, 0x72, 0x75, 0x6d, 0x20, 0x61, 0x74, 0x2c, 0x20, 0x74,
  304. 0x72, 0x69, 0x73, 0x74, 0x69, 0x71, 0x75, 0x65, 0x20, 0x6e, 0x6f, 0x6e, 0x20, 0x6e,
  305. 0x75, 0x6e, 0x63, 0x20, 0x65, 0x72, 0x61, 0x74, 0x20, 0x63, 0x75, 0x72, 0x61, 0x65,
  306. 0x2e, 0x20, 0x0a,
  307. ];
  308. let ieee802154_frame = Ieee802154Frame::new_checked(frame3).unwrap();
  309. let ieee802154_repr = Ieee802154Repr::parse(&ieee802154_frame).unwrap();
  310. let sixlowpan_frame =
  311. SixlowpanPacket::dispatch(ieee802154_frame.payload().unwrap()).unwrap();
  312. let frag = if let SixlowpanPacket::FragmentHeader = sixlowpan_frame {
  313. frag::Packet::new_checked(ieee802154_frame.payload().unwrap()).unwrap()
  314. } else {
  315. unreachable!()
  316. };
  317. assert_eq!(frag.datagram_size(), 307);
  318. assert_eq!(frag.datagram_tag(), 0x003f);
  319. assert_eq!(frag.datagram_offset(), 232 / 8);
  320. assert_eq!(frag.get_key(&ieee802154_repr), key);
  321. }
  322. }