sixlowpan.rs 89 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463
  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. const ADDRESS_CONTEXT_LENGTH: usize = 8;
  10. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  11. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  12. pub struct AddressContext(pub [u8; ADDRESS_CONTEXT_LENGTH]);
  13. /// The representation of an unresolved address. 6LoWPAN compression of IPv6 addresses can be with
  14. /// and without context information. The decompression with context information is not yet
  15. /// implemented.
  16. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  17. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  18. pub enum UnresolvedAddress<'a> {
  19. WithoutContext(AddressMode<'a>),
  20. WithContext((usize, AddressMode<'a>)),
  21. Reserved,
  22. }
  23. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  24. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  25. pub enum AddressMode<'a> {
  26. /// The full address is carried in-line.
  27. FullInline(&'a [u8]),
  28. /// The first 64-bits of the address are elided. The value of those bits
  29. /// is the link-local prefix padded with zeros. The remaining 64 bits are
  30. /// carried in-line.
  31. InLine64bits(&'a [u8]),
  32. /// The first 112 bits of the address are elided. The value of the first
  33. /// 64 bits is the link-local prefix padded with zeros. The following 64 bits
  34. /// are 0000:00ff:fe00:XXXX, where XXXX are the 16 bits carried in-line.
  35. InLine16bits(&'a [u8]),
  36. /// The address is fully elided. The first 64 bits of the address are
  37. /// the link-local prefix padded with zeros. The remaining 64 bits are
  38. /// computed from the encapsulating header (e.g., 802.15.4 or IPv6 source address)
  39. /// as specified in Section 3.2.2.
  40. FullyElided,
  41. /// The address takes the form ffXX::00XX:XXXX:XXXX
  42. Multicast48bits(&'a [u8]),
  43. /// The address takes the form ffXX::00XX:XXXX.
  44. Multicast32bits(&'a [u8]),
  45. /// The address takes the form ff02::00XX.
  46. Multicast8bits(&'a [u8]),
  47. /// The unspecified address.
  48. Unspecified,
  49. NotSupported,
  50. }
  51. const LINK_LOCAL_PREFIX: [u8; 2] = [0xfe, 0x80];
  52. const EUI64_MIDDLE_VALUE: [u8; 2] = [0xff, 0xfe];
  53. impl<'a> UnresolvedAddress<'a> {
  54. pub fn resolve(
  55. self,
  56. ll_address: Option<LlAddress>,
  57. addr_context: &[AddressContext],
  58. ) -> Result<ipv6::Address> {
  59. let mut bytes = [0; 16];
  60. let copy_context = |index: usize, bytes: &mut [u8]| -> Result<()> {
  61. if index >= addr_context.len() {
  62. return Err(Error);
  63. }
  64. let context = addr_context[index];
  65. bytes[..ADDRESS_CONTEXT_LENGTH].copy_from_slice(&context.0);
  66. Ok(())
  67. };
  68. match self {
  69. UnresolvedAddress::WithoutContext(mode) => match mode {
  70. AddressMode::FullInline(addr) => Ok(ipv6::Address::from_bytes(addr)),
  71. AddressMode::InLine64bits(inline) => {
  72. bytes[0..2].copy_from_slice(&LINK_LOCAL_PREFIX[..]);
  73. bytes[8..].copy_from_slice(inline);
  74. Ok(ipv6::Address::from_bytes(&bytes[..]))
  75. }
  76. AddressMode::InLine16bits(inline) => {
  77. bytes[0..2].copy_from_slice(&LINK_LOCAL_PREFIX[..]);
  78. bytes[11..13].copy_from_slice(&EUI64_MIDDLE_VALUE[..]);
  79. bytes[14..].copy_from_slice(inline);
  80. Ok(ipv6::Address::from_bytes(&bytes[..]))
  81. }
  82. AddressMode::FullyElided => {
  83. bytes[0..2].copy_from_slice(&LINK_LOCAL_PREFIX[..]);
  84. match ll_address {
  85. Some(LlAddress::Short(ll)) => {
  86. bytes[11..13].copy_from_slice(&EUI64_MIDDLE_VALUE[..]);
  87. bytes[14..].copy_from_slice(&ll);
  88. }
  89. Some(addr @ LlAddress::Extended(_)) => match addr.as_eui_64() {
  90. Some(addr) => bytes[8..].copy_from_slice(&addr),
  91. None => return Err(Error),
  92. },
  93. Some(LlAddress::Absent) => return Err(Error),
  94. None => return Err(Error),
  95. }
  96. Ok(ipv6::Address::from_bytes(&bytes[..]))
  97. }
  98. AddressMode::Multicast48bits(inline) => {
  99. bytes[0] = 0xff;
  100. bytes[1] = inline[0];
  101. bytes[11..].copy_from_slice(&inline[1..][..5]);
  102. Ok(ipv6::Address::from_bytes(&bytes[..]))
  103. }
  104. AddressMode::Multicast32bits(inline) => {
  105. bytes[0] = 0xff;
  106. bytes[1] = inline[0];
  107. bytes[13..].copy_from_slice(&inline[1..][..3]);
  108. Ok(ipv6::Address::from_bytes(&bytes[..]))
  109. }
  110. AddressMode::Multicast8bits(inline) => {
  111. bytes[0] = 0xff;
  112. bytes[1] = 0x02;
  113. bytes[15] = inline[0];
  114. Ok(ipv6::Address::from_bytes(&bytes[..]))
  115. }
  116. _ => Err(Error),
  117. },
  118. UnresolvedAddress::WithContext(mode) => match mode {
  119. (_, AddressMode::Unspecified) => Ok(ipv6::Address::UNSPECIFIED),
  120. (index, AddressMode::InLine64bits(inline)) => {
  121. copy_context(index, &mut bytes[..])?;
  122. bytes[16 - inline.len()..].copy_from_slice(inline);
  123. Ok(ipv6::Address::from_bytes(&bytes[..]))
  124. }
  125. (index, AddressMode::InLine16bits(inline)) => {
  126. copy_context(index, &mut bytes[..])?;
  127. bytes[16 - inline.len()..].copy_from_slice(inline);
  128. Ok(ipv6::Address::from_bytes(&bytes[..]))
  129. }
  130. (index, AddressMode::FullyElided) => {
  131. match ll_address {
  132. Some(LlAddress::Short(ll)) => {
  133. bytes[11..13].copy_from_slice(&EUI64_MIDDLE_VALUE[..]);
  134. bytes[14..].copy_from_slice(&ll);
  135. }
  136. Some(addr @ LlAddress::Extended(_)) => match addr.as_eui_64() {
  137. Some(addr) => bytes[8..].copy_from_slice(&addr),
  138. None => return Err(Error),
  139. },
  140. Some(LlAddress::Absent) => return Err(Error),
  141. None => return Err(Error),
  142. }
  143. copy_context(index, &mut bytes[..])?;
  144. Ok(ipv6::Address::from_bytes(&bytes[..]))
  145. }
  146. _ => Err(Error),
  147. },
  148. UnresolvedAddress::Reserved => Err(Error),
  149. }
  150. }
  151. }
  152. #[derive(Debug, Clone, Copy, PartialEq, Eq)]
  153. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  154. pub enum SixlowpanPacket {
  155. FragmentHeader,
  156. IphcHeader,
  157. }
  158. const DISPATCH_FIRST_FRAGMENT_HEADER: u8 = 0b11000;
  159. const DISPATCH_FRAGMENT_HEADER: u8 = 0b11100;
  160. const DISPATCH_IPHC_HEADER: u8 = 0b011;
  161. const DISPATCH_UDP_HEADER: u8 = 0b11110;
  162. const DISPATCH_EXT_HEADER: u8 = 0b1110;
  163. impl SixlowpanPacket {
  164. /// Returns the type of the 6LoWPAN header.
  165. /// This can either be a fragment header or an IPHC header.
  166. ///
  167. /// # Errors
  168. /// Returns `[Error::Unrecognized]` when neither the Fragment Header dispatch or the IPHC
  169. /// dispatch is recognized.
  170. pub fn dispatch(buffer: impl AsRef<[u8]>) -> Result<Self> {
  171. let raw = buffer.as_ref();
  172. if raw.is_empty() {
  173. return Err(Error);
  174. }
  175. if raw[0] >> 3 == DISPATCH_FIRST_FRAGMENT_HEADER || raw[0] >> 3 == DISPATCH_FRAGMENT_HEADER
  176. {
  177. Ok(Self::FragmentHeader)
  178. } else if raw[0] >> 5 == DISPATCH_IPHC_HEADER {
  179. Ok(Self::IphcHeader)
  180. } else {
  181. Err(Error)
  182. }
  183. }
  184. }
  185. pub mod frag {
  186. //! Implementation of the fragment headers from [RFC 4944 § 5.3].
  187. //!
  188. //! [RFC 4944 § 5.3]: https://datatracker.ietf.org/doc/html/rfc4944#section-5.3
  189. use super::{DISPATCH_FIRST_FRAGMENT_HEADER, DISPATCH_FRAGMENT_HEADER};
  190. use crate::wire::{Error, Result};
  191. use crate::wire::{Ieee802154Address, Ieee802154Repr};
  192. use byteorder::{ByteOrder, NetworkEndian};
  193. /// Key used for identifying all the link fragments that belong to the same packet.
  194. #[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
  195. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  196. pub struct Key {
  197. pub(crate) ll_src_addr: Ieee802154Address,
  198. pub(crate) ll_dst_addr: Ieee802154Address,
  199. pub(crate) datagram_size: u16,
  200. pub(crate) datagram_tag: u16,
  201. }
  202. /// A read/write wrapper around a 6LoWPAN Fragment header.
  203. /// [RFC 4944 § 5.3] specifies the format of the header.
  204. ///
  205. /// A First Fragment header has the following format:
  206. /// ```txt
  207. /// 1 2 3
  208. /// 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
  209. /// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  210. /// |1 1 0 0 0| datagram_size | datagram_tag |
  211. /// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  212. /// ```
  213. ///
  214. /// Subsequent fragment headers have the following format:
  215. /// ```txt
  216. /// 1 2 3
  217. /// 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
  218. /// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  219. /// |1 1 1 0 0| datagram_size | datagram_tag |
  220. /// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  221. /// |datagram_offset|
  222. /// +-+-+-+-+-+-+-+-+
  223. /// ```
  224. ///
  225. /// [RFC 4944 § 5.3]: https://datatracker.ietf.org/doc/html/rfc4944#section-5.3
  226. #[derive(Debug, Clone)]
  227. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  228. pub struct Packet<T: AsRef<[u8]>> {
  229. buffer: T,
  230. }
  231. pub const FIRST_FRAGMENT_HEADER_SIZE: usize = 4;
  232. pub const NEXT_FRAGMENT_HEADER_SIZE: usize = 5;
  233. mod field {
  234. use crate::wire::field::*;
  235. pub const DISPATCH: usize = 0;
  236. pub const DATAGRAM_SIZE: Field = 0..2;
  237. pub const DATAGRAM_TAG: Field = 2..4;
  238. pub const DATAGRAM_OFFSET: usize = 4;
  239. pub const FIRST_FRAGMENT_REST: Rest = super::FIRST_FRAGMENT_HEADER_SIZE..;
  240. pub const NEXT_FRAGMENT_REST: Rest = super::NEXT_FRAGMENT_HEADER_SIZE..;
  241. }
  242. impl<T: AsRef<[u8]>> Packet<T> {
  243. /// Input a raw octet buffer with a 6LoWPAN Fragment header structure.
  244. pub const fn new_unchecked(buffer: T) -> Self {
  245. Self { buffer }
  246. }
  247. /// Shorthand for a combination of [new_unchecked] and [check_len].
  248. ///
  249. /// [new_unchecked]: #method.new_unchecked
  250. /// [check_len]: #method.check_len
  251. pub fn new_checked(buffer: T) -> Result<Self> {
  252. let packet = Self::new_unchecked(buffer);
  253. packet.check_len()?;
  254. let dispatch = packet.dispatch();
  255. if dispatch != DISPATCH_FIRST_FRAGMENT_HEADER && dispatch != DISPATCH_FRAGMENT_HEADER {
  256. return Err(Error);
  257. }
  258. Ok(packet)
  259. }
  260. /// Ensure that no accessor method will panic if called.
  261. /// Returns `Err(Error)` if the buffer is too short.
  262. pub fn check_len(&self) -> Result<()> {
  263. let buffer = self.buffer.as_ref();
  264. if buffer.is_empty() {
  265. return Err(Error);
  266. }
  267. match self.dispatch() {
  268. DISPATCH_FIRST_FRAGMENT_HEADER if buffer.len() >= FIRST_FRAGMENT_HEADER_SIZE => {
  269. Ok(())
  270. }
  271. DISPATCH_FIRST_FRAGMENT_HEADER if buffer.len() < FIRST_FRAGMENT_HEADER_SIZE => {
  272. Err(Error)
  273. }
  274. DISPATCH_FRAGMENT_HEADER if buffer.len() >= NEXT_FRAGMENT_HEADER_SIZE => Ok(()),
  275. DISPATCH_FRAGMENT_HEADER if buffer.len() < NEXT_FRAGMENT_HEADER_SIZE => Err(Error),
  276. _ => Err(Error),
  277. }
  278. }
  279. /// Consumes the frame, returning the underlying buffer.
  280. pub fn into_inner(self) -> T {
  281. self.buffer
  282. }
  283. /// Return the dispatch field.
  284. pub fn dispatch(&self) -> u8 {
  285. let raw = self.buffer.as_ref();
  286. raw[field::DISPATCH] >> 3
  287. }
  288. /// Return the total datagram size.
  289. pub fn datagram_size(&self) -> u16 {
  290. let raw = self.buffer.as_ref();
  291. NetworkEndian::read_u16(&raw[field::DATAGRAM_SIZE]) & 0b111_1111_1111
  292. }
  293. /// Return the datagram tag.
  294. pub fn datagram_tag(&self) -> u16 {
  295. let raw = self.buffer.as_ref();
  296. NetworkEndian::read_u16(&raw[field::DATAGRAM_TAG])
  297. }
  298. /// Return the datagram offset.
  299. pub fn datagram_offset(&self) -> u8 {
  300. match self.dispatch() {
  301. DISPATCH_FIRST_FRAGMENT_HEADER => 0,
  302. DISPATCH_FRAGMENT_HEADER => {
  303. let raw = self.buffer.as_ref();
  304. raw[field::DATAGRAM_OFFSET]
  305. }
  306. _ => unreachable!(),
  307. }
  308. }
  309. /// Returns `true` when this header is from the first fragment of a link.
  310. pub fn is_first_fragment(&self) -> bool {
  311. self.dispatch() == DISPATCH_FIRST_FRAGMENT_HEADER
  312. }
  313. /// Returns the key for identifying the packet it belongs to.
  314. pub fn get_key(&self, ieee802154_repr: &Ieee802154Repr) -> Key {
  315. Key {
  316. ll_src_addr: ieee802154_repr.src_addr.unwrap(),
  317. ll_dst_addr: ieee802154_repr.dst_addr.unwrap(),
  318. datagram_size: self.datagram_size(),
  319. datagram_tag: self.datagram_tag(),
  320. }
  321. }
  322. }
  323. impl<'a, T: AsRef<[u8]> + ?Sized> Packet<&'a T> {
  324. /// Return the payload.
  325. pub fn payload(&self) -> &'a [u8] {
  326. match self.dispatch() {
  327. DISPATCH_FIRST_FRAGMENT_HEADER => {
  328. let raw = self.buffer.as_ref();
  329. &raw[field::FIRST_FRAGMENT_REST]
  330. }
  331. DISPATCH_FRAGMENT_HEADER => {
  332. let raw = self.buffer.as_ref();
  333. &raw[field::NEXT_FRAGMENT_REST]
  334. }
  335. _ => unreachable!(),
  336. }
  337. }
  338. }
  339. impl<T: AsRef<[u8]> + AsMut<[u8]>> Packet<T> {
  340. fn set_dispatch_field(&mut self, value: u8) {
  341. let raw = self.buffer.as_mut();
  342. raw[field::DISPATCH] = (raw[field::DISPATCH] & !(0b11111 << 3)) | (value << 3);
  343. }
  344. fn set_datagram_size(&mut self, size: u16) {
  345. let raw = self.buffer.as_mut();
  346. let mut v = NetworkEndian::read_u16(&raw[field::DATAGRAM_SIZE]);
  347. v = (v & !0b111_1111_1111) | size;
  348. NetworkEndian::write_u16(&mut raw[field::DATAGRAM_SIZE], v);
  349. }
  350. fn set_datagram_tag(&mut self, tag: u16) {
  351. let raw = self.buffer.as_mut();
  352. NetworkEndian::write_u16(&mut raw[field::DATAGRAM_TAG], tag);
  353. }
  354. fn set_datagram_offset(&mut self, offset: u8) {
  355. let raw = self.buffer.as_mut();
  356. raw[field::DATAGRAM_OFFSET] = offset;
  357. }
  358. }
  359. /// A high-level representation of a 6LoWPAN Fragment header.
  360. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  361. pub enum Repr {
  362. FirstFragment { size: u16, tag: u16 },
  363. Fragment { size: u16, tag: u16, offset: u8 },
  364. }
  365. impl core::fmt::Display for Repr {
  366. fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
  367. match self {
  368. Repr::FirstFragment { size, tag } => {
  369. write!(f, "FirstFrag size={size} tag={tag}")
  370. }
  371. Repr::Fragment { size, tag, offset } => {
  372. write!(f, "NthFrag size={size} tag={tag} offset={offset}")
  373. }
  374. }
  375. }
  376. }
  377. #[cfg(feature = "defmt")]
  378. impl defmt::Format for Repr {
  379. fn format(&self, fmt: defmt::Formatter) {
  380. match self {
  381. Repr::FirstFragment { size, tag } => {
  382. defmt::write!(fmt, "FirstFrag size={} tag={}", size, tag);
  383. }
  384. Repr::Fragment { size, tag, offset } => {
  385. defmt::write!(fmt, "NthFrag size={} tag={} offset={}", size, tag, offset);
  386. }
  387. }
  388. }
  389. }
  390. impl Repr {
  391. /// Parse a 6LoWPAN Fragment header.
  392. pub fn parse<T: AsRef<[u8]>>(packet: &Packet<T>) -> Result<Self> {
  393. let size = packet.datagram_size();
  394. let tag = packet.datagram_tag();
  395. match packet.dispatch() {
  396. DISPATCH_FIRST_FRAGMENT_HEADER => Ok(Self::FirstFragment { size, tag }),
  397. DISPATCH_FRAGMENT_HEADER => Ok(Self::Fragment {
  398. size,
  399. tag,
  400. offset: packet.datagram_offset(),
  401. }),
  402. _ => Err(Error),
  403. }
  404. }
  405. /// Returns the length of the Fragment header.
  406. pub const fn buffer_len(&self) -> usize {
  407. match self {
  408. Self::FirstFragment { .. } => field::FIRST_FRAGMENT_REST.start,
  409. Self::Fragment { .. } => field::NEXT_FRAGMENT_REST.start,
  410. }
  411. }
  412. /// Emit a high-level representation into a 6LoWPAN Fragment header.
  413. pub fn emit<T: AsRef<[u8]> + AsMut<[u8]>>(&self, packet: &mut Packet<T>) {
  414. match self {
  415. Self::FirstFragment { size, tag } => {
  416. packet.set_dispatch_field(DISPATCH_FIRST_FRAGMENT_HEADER);
  417. packet.set_datagram_size(*size);
  418. packet.set_datagram_tag(*tag);
  419. }
  420. Self::Fragment { size, tag, offset } => {
  421. packet.set_dispatch_field(DISPATCH_FRAGMENT_HEADER);
  422. packet.set_datagram_size(*size);
  423. packet.set_datagram_tag(*tag);
  424. packet.set_datagram_offset(*offset);
  425. }
  426. }
  427. }
  428. }
  429. }
  430. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  431. pub enum NextHeader {
  432. Compressed,
  433. Uncompressed(IpProtocol),
  434. }
  435. impl core::fmt::Display for NextHeader {
  436. fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
  437. match self {
  438. NextHeader::Compressed => write!(f, "compressed"),
  439. NextHeader::Uncompressed(protocol) => write!(f, "{protocol}"),
  440. }
  441. }
  442. }
  443. #[cfg(feature = "defmt")]
  444. impl defmt::Format for NextHeader {
  445. fn format(&self, fmt: defmt::Formatter) {
  446. match self {
  447. NextHeader::Compressed => defmt::write!(fmt, "compressed"),
  448. NextHeader::Uncompressed(protocol) => defmt::write!(fmt, "{}", protocol),
  449. }
  450. }
  451. }
  452. pub mod iphc {
  453. //! Implementation of IP Header Compression from [RFC 6282 § 3.1].
  454. //! It defines the compression of IPv6 headers.
  455. //!
  456. //! [RFC 6282 § 3.1]: https://datatracker.ietf.org/doc/html/rfc6282#section-3.1
  457. use super::{
  458. AddressContext, AddressMode, Error, NextHeader, Result, UnresolvedAddress,
  459. DISPATCH_IPHC_HEADER,
  460. };
  461. use crate::wire::{ieee802154::Address as LlAddress, ipv6, IpProtocol};
  462. use byteorder::{ByteOrder, NetworkEndian};
  463. mod field {
  464. use crate::wire::field::*;
  465. pub const IPHC_FIELD: Field = 0..2;
  466. }
  467. macro_rules! get_field {
  468. ($name:ident, $mask:expr, $shift:expr) => {
  469. fn $name(&self) -> u8 {
  470. let data = self.buffer.as_ref();
  471. let raw = NetworkEndian::read_u16(&data[field::IPHC_FIELD]);
  472. ((raw >> $shift) & $mask) as u8
  473. }
  474. };
  475. }
  476. macro_rules! set_field {
  477. ($name:ident, $mask:expr, $shift:expr) => {
  478. fn $name(&mut self, val: u8) {
  479. let data = &mut self.buffer.as_mut()[field::IPHC_FIELD];
  480. let mut raw = NetworkEndian::read_u16(data);
  481. raw = (raw & !($mask << $shift)) | ((val as u16) << $shift);
  482. NetworkEndian::write_u16(data, raw);
  483. }
  484. };
  485. }
  486. /// A read/write wrapper around a 6LoWPAN IPHC header.
  487. /// [RFC 6282 § 3.1] specifies the format of the header.
  488. ///
  489. /// The header always start with the following base format (from [RFC 6282 § 3.1.1]):
  490. /// ```txt
  491. /// 0 1
  492. /// 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
  493. /// +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
  494. /// | 0 | 1 | 1 | TF |NH | HLIM |CID|SAC| SAM | M |DAC| DAM |
  495. /// +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+
  496. /// ```
  497. /// With:
  498. /// - TF: Traffic Class and Flow Label
  499. /// - NH: Next Header
  500. /// - HLIM: Hop Limit
  501. /// - CID: Context Identifier Extension
  502. /// - SAC: Source Address Compression
  503. /// - SAM: Source Address Mode
  504. /// - M: Multicast Compression
  505. /// - DAC: Destination Address Compression
  506. /// - DAM: Destination Address Mode
  507. ///
  508. /// Depending on the flags in the base format, the following fields are added to the header:
  509. /// - Traffic Class and Flow Label
  510. /// - Next Header
  511. /// - Hop Limit
  512. /// - IPv6 source address
  513. /// - IPv6 destinatino address
  514. ///
  515. /// [RFC 6282 § 3.1]: https://datatracker.ietf.org/doc/html/rfc6282#section-3.1
  516. /// [RFC 6282 § 3.1.1]: https://datatracker.ietf.org/doc/html/rfc6282#section-3.1.1
  517. #[derive(Debug, Clone)]
  518. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  519. pub struct Packet<T: AsRef<[u8]>> {
  520. buffer: T,
  521. }
  522. impl<T: AsRef<[u8]>> Packet<T> {
  523. /// Input a raw octet buffer with a 6LoWPAN IPHC header structure.
  524. pub const fn new_unchecked(buffer: T) -> Self {
  525. Packet { buffer }
  526. }
  527. /// Shorthand for a combination of [new_unchecked] and [check_len].
  528. ///
  529. /// [new_unchecked]: #method.new_unchecked
  530. /// [check_len]: #method.check_len
  531. pub fn new_checked(buffer: T) -> Result<Self> {
  532. let packet = Self::new_unchecked(buffer);
  533. packet.check_len()?;
  534. Ok(packet)
  535. }
  536. /// Ensure that no accessor method will panic if called.
  537. /// Returns `Err(Error)` if the buffer is too short.
  538. pub fn check_len(&self) -> Result<()> {
  539. let buffer = self.buffer.as_ref();
  540. if buffer.len() < 2 {
  541. return Err(Error);
  542. }
  543. let mut offset = self.ip_fields_start()
  544. + self.traffic_class_size()
  545. + self.next_header_size()
  546. + self.hop_limit_size();
  547. offset += self.src_address_size();
  548. offset += self.dst_address_size();
  549. if offset as usize > buffer.len() {
  550. return Err(Error);
  551. }
  552. Ok(())
  553. }
  554. /// Consumes the frame, returning the underlying buffer.
  555. pub fn into_inner(self) -> T {
  556. self.buffer
  557. }
  558. /// Return the Next Header field.
  559. pub fn next_header(&self) -> NextHeader {
  560. let nh = self.nh_field();
  561. if nh == 1 {
  562. // The next header field is compressed.
  563. // It is also encoded using LOWPAN_NHC.
  564. NextHeader::Compressed
  565. } else {
  566. // The full 8 bits for Next Header are carried in-line.
  567. let start = (self.ip_fields_start() + self.traffic_class_size()) as usize;
  568. let data = self.buffer.as_ref();
  569. let nh = data[start..start + 1][0];
  570. NextHeader::Uncompressed(IpProtocol::from(nh))
  571. }
  572. }
  573. /// Return the Hop Limit.
  574. pub fn hop_limit(&self) -> u8 {
  575. match self.hlim_field() {
  576. 0b00 => {
  577. let start = (self.ip_fields_start()
  578. + self.traffic_class_size()
  579. + self.next_header_size()) as usize;
  580. let data = self.buffer.as_ref();
  581. data[start..start + 1][0]
  582. }
  583. 0b01 => 1,
  584. 0b10 => 64,
  585. 0b11 => 255,
  586. _ => unreachable!(),
  587. }
  588. }
  589. /// Return the Source Context Identifier.
  590. pub fn src_context_id(&self) -> Option<u8> {
  591. if self.cid_field() == 1 {
  592. let data = self.buffer.as_ref();
  593. Some(data[2] >> 4)
  594. } else {
  595. None
  596. }
  597. }
  598. /// Return the Destination Context Identifier.
  599. pub fn dst_context_id(&self) -> Option<u8> {
  600. if self.cid_field() == 1 {
  601. let data = self.buffer.as_ref();
  602. Some(data[2] & 0x0f)
  603. } else {
  604. None
  605. }
  606. }
  607. /// Return the ECN field (when it is inlined).
  608. pub fn ecn_field(&self) -> Option<u8> {
  609. match self.tf_field() {
  610. 0b00 | 0b01 | 0b10 => {
  611. let start = self.ip_fields_start() as usize;
  612. Some(self.buffer.as_ref()[start..][0] & 0b1100_0000)
  613. }
  614. 0b11 => None,
  615. _ => unreachable!(),
  616. }
  617. }
  618. /// Return the DSCP field (when it is inlined).
  619. pub fn dscp_field(&self) -> Option<u8> {
  620. match self.tf_field() {
  621. 0b00 | 0b10 => {
  622. let start = self.ip_fields_start() as usize;
  623. Some(self.buffer.as_ref()[start..][0] & 0b111111)
  624. }
  625. 0b01 | 0b11 => None,
  626. _ => unreachable!(),
  627. }
  628. }
  629. /// Return the flow label field (when it is inlined).
  630. pub fn flow_label_field(&self) -> Option<u16> {
  631. match self.tf_field() {
  632. 0b00 => {
  633. let start = self.ip_fields_start() as usize;
  634. Some(NetworkEndian::read_u16(
  635. &self.buffer.as_ref()[start..][2..4],
  636. ))
  637. }
  638. 0b01 => {
  639. let start = self.ip_fields_start() as usize;
  640. Some(NetworkEndian::read_u16(
  641. &self.buffer.as_ref()[start..][1..3],
  642. ))
  643. }
  644. 0b10 | 0b11 => None,
  645. _ => unreachable!(),
  646. }
  647. }
  648. /// Return the Source Address.
  649. pub fn src_addr(&self) -> Result<UnresolvedAddress> {
  650. let start = (self.ip_fields_start()
  651. + self.traffic_class_size()
  652. + self.next_header_size()
  653. + self.hop_limit_size()) as usize;
  654. let data = self.buffer.as_ref();
  655. match (self.sac_field(), self.sam_field()) {
  656. (0, 0b00) => Ok(UnresolvedAddress::WithoutContext(AddressMode::FullInline(
  657. &data[start..][..16],
  658. ))),
  659. (0, 0b01) => Ok(UnresolvedAddress::WithoutContext(
  660. AddressMode::InLine64bits(&data[start..][..8]),
  661. )),
  662. (0, 0b10) => Ok(UnresolvedAddress::WithoutContext(
  663. AddressMode::InLine16bits(&data[start..][..2]),
  664. )),
  665. (0, 0b11) => Ok(UnresolvedAddress::WithoutContext(AddressMode::FullyElided)),
  666. (1, 0b00) => Ok(UnresolvedAddress::WithContext((
  667. 0,
  668. AddressMode::Unspecified,
  669. ))),
  670. (1, 0b01) => {
  671. if let Some(id) = self.src_context_id() {
  672. Ok(UnresolvedAddress::WithContext((
  673. id as usize,
  674. AddressMode::InLine64bits(&data[start..][..8]),
  675. )))
  676. } else {
  677. Err(Error)
  678. }
  679. }
  680. (1, 0b10) => {
  681. if let Some(id) = self.src_context_id() {
  682. Ok(UnresolvedAddress::WithContext((
  683. id as usize,
  684. AddressMode::InLine16bits(&data[start..][..2]),
  685. )))
  686. } else {
  687. Err(Error)
  688. }
  689. }
  690. (1, 0b11) => {
  691. if let Some(id) = self.src_context_id() {
  692. Ok(UnresolvedAddress::WithContext((
  693. id as usize,
  694. AddressMode::FullyElided,
  695. )))
  696. } else {
  697. Err(Error)
  698. }
  699. }
  700. _ => Err(Error),
  701. }
  702. }
  703. /// Return the Destination Address.
  704. pub fn dst_addr(&self) -> Result<UnresolvedAddress> {
  705. let start = (self.ip_fields_start()
  706. + self.traffic_class_size()
  707. + self.next_header_size()
  708. + self.hop_limit_size()
  709. + self.src_address_size()) as usize;
  710. let data = self.buffer.as_ref();
  711. match (self.m_field(), self.dac_field(), self.dam_field()) {
  712. (0, 0, 0b00) => Ok(UnresolvedAddress::WithoutContext(AddressMode::FullInline(
  713. &data[start..][..16],
  714. ))),
  715. (0, 0, 0b01) => Ok(UnresolvedAddress::WithoutContext(
  716. AddressMode::InLine64bits(&data[start..][..8]),
  717. )),
  718. (0, 0, 0b10) => Ok(UnresolvedAddress::WithoutContext(
  719. AddressMode::InLine16bits(&data[start..][..2]),
  720. )),
  721. (0, 0, 0b11) => Ok(UnresolvedAddress::WithoutContext(AddressMode::FullyElided)),
  722. (0, 1, 0b00) => Ok(UnresolvedAddress::Reserved),
  723. (0, 1, 0b01) => {
  724. if let Some(id) = self.dst_context_id() {
  725. Ok(UnresolvedAddress::WithContext((
  726. id as usize,
  727. AddressMode::InLine64bits(&data[start..][..8]),
  728. )))
  729. } else {
  730. Err(Error)
  731. }
  732. }
  733. (0, 1, 0b10) => {
  734. if let Some(id) = self.dst_context_id() {
  735. Ok(UnresolvedAddress::WithContext((
  736. id as usize,
  737. AddressMode::InLine16bits(&data[start..][..2]),
  738. )))
  739. } else {
  740. Err(Error)
  741. }
  742. }
  743. (0, 1, 0b11) => {
  744. if let Some(id) = self.dst_context_id() {
  745. Ok(UnresolvedAddress::WithContext((
  746. id as usize,
  747. AddressMode::FullyElided,
  748. )))
  749. } else {
  750. Err(Error)
  751. }
  752. }
  753. (1, 0, 0b00) => Ok(UnresolvedAddress::WithoutContext(AddressMode::FullInline(
  754. &data[start..][..16],
  755. ))),
  756. (1, 0, 0b01) => Ok(UnresolvedAddress::WithoutContext(
  757. AddressMode::Multicast48bits(&data[start..][..6]),
  758. )),
  759. (1, 0, 0b10) => Ok(UnresolvedAddress::WithoutContext(
  760. AddressMode::Multicast32bits(&data[start..][..4]),
  761. )),
  762. (1, 0, 0b11) => Ok(UnresolvedAddress::WithoutContext(
  763. AddressMode::Multicast8bits(&data[start..][..1]),
  764. )),
  765. (1, 1, 0b00) => Ok(UnresolvedAddress::WithContext((
  766. 0,
  767. AddressMode::NotSupported,
  768. ))),
  769. (1, 1, 0b01 | 0b10 | 0b11) => Ok(UnresolvedAddress::Reserved),
  770. _ => Err(Error),
  771. }
  772. }
  773. get_field!(dispatch_field, 0b111, 13);
  774. get_field!(tf_field, 0b11, 11);
  775. get_field!(nh_field, 0b1, 10);
  776. get_field!(hlim_field, 0b11, 8);
  777. get_field!(cid_field, 0b1, 7);
  778. get_field!(sac_field, 0b1, 6);
  779. get_field!(sam_field, 0b11, 4);
  780. get_field!(m_field, 0b1, 3);
  781. get_field!(dac_field, 0b1, 2);
  782. get_field!(dam_field, 0b11, 0);
  783. /// Return the start for the IP fields.
  784. fn ip_fields_start(&self) -> u8 {
  785. 2 + self.cid_size()
  786. }
  787. /// Get the size in octets of the traffic class field.
  788. fn traffic_class_size(&self) -> u8 {
  789. match self.tf_field() {
  790. 0b00 => 4,
  791. 0b01 => 3,
  792. 0b10 => 1,
  793. 0b11 => 0,
  794. _ => unreachable!(),
  795. }
  796. }
  797. /// Get the size in octets of the next header field.
  798. fn next_header_size(&self) -> u8 {
  799. (self.nh_field() != 1) as u8
  800. }
  801. /// Get the size in octets of the hop limit field.
  802. fn hop_limit_size(&self) -> u8 {
  803. (self.hlim_field() == 0b00) as u8
  804. }
  805. /// Get the size in octets of the CID field.
  806. fn cid_size(&self) -> u8 {
  807. (self.cid_field() == 1) as u8
  808. }
  809. /// Get the size in octets of the source address.
  810. fn src_address_size(&self) -> u8 {
  811. match (self.sac_field(), self.sam_field()) {
  812. (0, 0b00) => 16, // The full address is carried in-line.
  813. (0, 0b01) => 8, // The first 64 bits are elided.
  814. (0, 0b10) => 2, // The first 112 bits are elided.
  815. (0, 0b11) => 0, // The address is fully elided.
  816. (1, 0b00) => 0, // The UNSPECIFIED address.
  817. (1, 0b01) => 8, // Address derived using context information.
  818. (1, 0b10) => 2, // Address derived using context information.
  819. (1, 0b11) => 0, // Address derived using context information.
  820. _ => unreachable!(),
  821. }
  822. }
  823. /// Get the size in octets of the address address.
  824. fn dst_address_size(&self) -> u8 {
  825. match (self.m_field(), self.dac_field(), self.dam_field()) {
  826. (0, 0, 0b00) => 16, // The full address is carried in-line.
  827. (0, 0, 0b01) => 8, // The first 64 bits are elided.
  828. (0, 0, 0b10) => 2, // The first 112 bits are elided.
  829. (0, 0, 0b11) => 0, // The address is fully elided.
  830. (0, 1, 0b00) => 0, // Reserved.
  831. (0, 1, 0b01) => 8, // Address derived using context information.
  832. (0, 1, 0b10) => 2, // Address derived using context information.
  833. (0, 1, 0b11) => 0, // Address derived using context information.
  834. (1, 0, 0b00) => 16, // The full address is carried in-line.
  835. (1, 0, 0b01) => 6, // The address takes the form ffXX::00XX:XXXX:XXXX.
  836. (1, 0, 0b10) => 4, // The address takes the form ffXX::00XX:XXXX.
  837. (1, 0, 0b11) => 1, // The address takes the form ff02::00XX.
  838. (1, 1, 0b00) => 6, // Match Unicast-Prefix-based IPv6.
  839. (1, 1, 0b01) => 0, // Reserved.
  840. (1, 1, 0b10) => 0, // Reserved.
  841. (1, 1, 0b11) => 0, // Reserved.
  842. _ => unreachable!(),
  843. }
  844. }
  845. /// Return the length of the header.
  846. pub fn header_len(&self) -> usize {
  847. let mut len = self.ip_fields_start();
  848. len += self.traffic_class_size();
  849. len += self.next_header_size();
  850. len += self.hop_limit_size();
  851. len += self.src_address_size();
  852. len += self.dst_address_size();
  853. len as usize
  854. }
  855. }
  856. impl<'a, T: AsRef<[u8]> + ?Sized> Packet<&'a T> {
  857. /// Return a pointer to the payload.
  858. pub fn payload(&self) -> &'a [u8] {
  859. let mut len = self.ip_fields_start();
  860. len += self.traffic_class_size();
  861. len += self.next_header_size();
  862. len += self.hop_limit_size();
  863. len += self.src_address_size();
  864. len += self.dst_address_size();
  865. let len = len as usize;
  866. let data = self.buffer.as_ref();
  867. &data[len..]
  868. }
  869. }
  870. impl<T: AsRef<[u8]> + AsMut<[u8]>> Packet<T> {
  871. /// Set the dispatch field to `0b011`.
  872. fn set_dispatch_field(&mut self) {
  873. let data = &mut self.buffer.as_mut()[field::IPHC_FIELD];
  874. let mut raw = NetworkEndian::read_u16(data);
  875. raw = (raw & !(0b111 << 13)) | (0b11 << 13);
  876. NetworkEndian::write_u16(data, raw);
  877. }
  878. set_field!(set_tf_field, 0b11, 11);
  879. set_field!(set_nh_field, 0b1, 10);
  880. set_field!(set_hlim_field, 0b11, 8);
  881. set_field!(set_cid_field, 0b1, 7);
  882. set_field!(set_sac_field, 0b1, 6);
  883. set_field!(set_sam_field, 0b11, 4);
  884. set_field!(set_m_field, 0b1, 3);
  885. set_field!(set_dac_field, 0b1, 2);
  886. set_field!(set_dam_field, 0b11, 0);
  887. fn set_field(&mut self, idx: usize, value: &[u8]) {
  888. let raw = self.buffer.as_mut();
  889. raw[idx..idx + value.len()].copy_from_slice(value);
  890. }
  891. /// Set the Next Header.
  892. ///
  893. /// **NOTE**: `idx` is the offset at which the Next Header needs to be written to.
  894. fn set_next_header(&mut self, nh: NextHeader, mut idx: usize) -> usize {
  895. match nh {
  896. NextHeader::Uncompressed(nh) => {
  897. self.set_nh_field(0);
  898. self.set_field(idx, &[nh.into()]);
  899. idx += 1;
  900. }
  901. NextHeader::Compressed => self.set_nh_field(1),
  902. }
  903. idx
  904. }
  905. /// Set the Hop Limit.
  906. ///
  907. /// **NOTE**: `idx` is the offset at which the Next Header needs to be written to.
  908. fn set_hop_limit(&mut self, hl: u8, mut idx: usize) -> usize {
  909. match hl {
  910. 255 => self.set_hlim_field(0b11),
  911. 64 => self.set_hlim_field(0b10),
  912. 1 => self.set_hlim_field(0b01),
  913. _ => {
  914. self.set_hlim_field(0b00);
  915. self.set_field(idx, &[hl]);
  916. idx += 1;
  917. }
  918. }
  919. idx
  920. }
  921. /// Set the Source Address based on the IPv6 address and the Link-Local address.
  922. ///
  923. /// **NOTE**: `idx` is the offset at which the Next Header needs to be written to.
  924. fn set_src_address(
  925. &mut self,
  926. src_addr: ipv6::Address,
  927. ll_src_addr: Option<LlAddress>,
  928. mut idx: usize,
  929. ) -> usize {
  930. self.set_cid_field(0);
  931. self.set_sac_field(0);
  932. let src = src_addr.as_bytes();
  933. if src_addr == ipv6::Address::UNSPECIFIED {
  934. self.set_sac_field(1);
  935. self.set_sam_field(0b00);
  936. } else if src_addr.is_link_local() {
  937. // We have a link local address.
  938. // The remainder of the address can be elided when the context contains
  939. // a 802.15.4 short address or a 802.15.4 extended address which can be
  940. // converted to a eui64 address.
  941. let is_eui_64 = ll_src_addr
  942. .map(|addr| {
  943. addr.as_eui_64()
  944. .map(|addr| addr[..] == src[8..])
  945. .unwrap_or(false)
  946. })
  947. .unwrap_or(false);
  948. if src[8..14] == [0, 0, 0, 0xff, 0xfe, 0] {
  949. let ll = [src[14], src[15]];
  950. if ll_src_addr == Some(LlAddress::Short(ll)) {
  951. // We have the context from the 802.15.4 frame.
  952. // The context contains the short address.
  953. // We can elide the source address.
  954. self.set_sam_field(0b11);
  955. } else {
  956. // We don't have the context from the 802.15.4 frame.
  957. // We cannot elide the source address, however we can elide 112 bits.
  958. self.set_sam_field(0b10);
  959. self.set_field(idx, &src[14..]);
  960. idx += 2;
  961. }
  962. } else if is_eui_64 {
  963. // We have the context from the 802.15.4 frame.
  964. // The context contains the extended address.
  965. // We can elide the source address.
  966. self.set_sam_field(0b11);
  967. } else {
  968. // We cannot elide the source address, however we can elide 64 bits.
  969. self.set_sam_field(0b01);
  970. self.set_field(idx, &src[8..]);
  971. idx += 8;
  972. }
  973. } else {
  974. // We cannot elide anything.
  975. self.set_sam_field(0b00);
  976. self.set_field(idx, src);
  977. idx += 16;
  978. }
  979. idx
  980. }
  981. /// Set the Destination Address based on the IPv6 address and the Link-Local address.
  982. ///
  983. /// **NOTE**: `idx` is the offset at which the Next Header needs to be written to.
  984. fn set_dst_address(
  985. &mut self,
  986. dst_addr: ipv6::Address,
  987. ll_dst_addr: Option<LlAddress>,
  988. mut idx: usize,
  989. ) -> usize {
  990. self.set_dac_field(0);
  991. self.set_dam_field(0);
  992. self.set_m_field(0);
  993. let dst = dst_addr.as_bytes();
  994. if dst_addr.is_multicast() {
  995. self.set_m_field(1);
  996. if dst[1] == 0x02 && dst[2..15] == [0; 13] {
  997. self.set_dam_field(0b11);
  998. self.set_field(idx, &[dst[15]]);
  999. idx += 1;
  1000. } else if dst[2..13] == [0; 11] {
  1001. self.set_dam_field(0b10);
  1002. self.set_field(idx, &[dst[1]]);
  1003. idx += 1;
  1004. self.set_field(idx, &dst[13..]);
  1005. idx += 3;
  1006. } else if dst[2..11] == [0; 9] {
  1007. self.set_dam_field(0b01);
  1008. self.set_field(idx, &[dst[1]]);
  1009. idx += 1;
  1010. self.set_field(idx, &dst[11..]);
  1011. idx += 5;
  1012. } else {
  1013. self.set_dam_field(0b11);
  1014. self.set_field(idx, dst);
  1015. idx += 16;
  1016. }
  1017. } else if dst_addr.is_link_local() {
  1018. let is_eui_64 = ll_dst_addr
  1019. .map(|addr| {
  1020. addr.as_eui_64()
  1021. .map(|addr| addr[..] == dst[8..])
  1022. .unwrap_or(false)
  1023. })
  1024. .unwrap_or(false);
  1025. if dst[8..14] == [0, 0, 0, 0xff, 0xfe, 0] {
  1026. let ll = [dst[14], dst[15]];
  1027. if ll_dst_addr == Some(LlAddress::Short(ll)) {
  1028. self.set_dam_field(0b11);
  1029. } else {
  1030. self.set_dam_field(0b10);
  1031. self.set_field(idx, &dst[14..]);
  1032. idx += 2;
  1033. }
  1034. } else if is_eui_64 {
  1035. self.set_dam_field(0b11);
  1036. } else {
  1037. self.set_dam_field(0b01);
  1038. self.set_field(idx, &dst[8..]);
  1039. idx += 8;
  1040. }
  1041. } else {
  1042. self.set_dam_field(0b00);
  1043. self.set_field(idx, dst);
  1044. idx += 16;
  1045. }
  1046. idx
  1047. }
  1048. /// Return a mutable pointer to the payload.
  1049. pub fn payload_mut(&mut self) -> &mut [u8] {
  1050. let mut len = self.ip_fields_start();
  1051. len += self.traffic_class_size();
  1052. len += self.next_header_size();
  1053. len += self.hop_limit_size();
  1054. len += self.src_address_size();
  1055. len += self.dst_address_size();
  1056. let len = len as usize;
  1057. let data = self.buffer.as_mut();
  1058. &mut data[len..]
  1059. }
  1060. }
  1061. /// A high-level representation of a 6LoWPAN IPHC header.
  1062. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  1063. pub struct Repr {
  1064. pub src_addr: ipv6::Address,
  1065. pub ll_src_addr: Option<LlAddress>,
  1066. pub dst_addr: ipv6::Address,
  1067. pub ll_dst_addr: Option<LlAddress>,
  1068. pub next_header: NextHeader,
  1069. pub hop_limit: u8,
  1070. // TODO(thvdveld): refactor the following fields into something else
  1071. pub ecn: Option<u8>,
  1072. pub dscp: Option<u8>,
  1073. pub flow_label: Option<u16>,
  1074. }
  1075. impl core::fmt::Display for Repr {
  1076. fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
  1077. write!(
  1078. f,
  1079. "IPHC src={} dst={} nxt-hdr={} hop-limit={}",
  1080. self.src_addr, self.dst_addr, self.next_header, self.hop_limit
  1081. )
  1082. }
  1083. }
  1084. #[cfg(feature = "defmt")]
  1085. impl defmt::Format for Repr {
  1086. fn format(&self, fmt: defmt::Formatter) {
  1087. defmt::write!(
  1088. fmt,
  1089. "IPHC src={} dst={} nxt-hdr={} hop-limit={}",
  1090. self.src_addr,
  1091. self.dst_addr,
  1092. self.next_header,
  1093. self.hop_limit
  1094. );
  1095. }
  1096. }
  1097. impl Repr {
  1098. /// Parse a 6LoWPAN IPHC header and return a high-level representation.
  1099. ///
  1100. /// The `ll_src_addr` and `ll_dst_addr` are the link-local addresses used for resolving the
  1101. /// IPv6 packets.
  1102. pub fn parse<T: AsRef<[u8]> + ?Sized>(
  1103. packet: &Packet<&T>,
  1104. ll_src_addr: Option<LlAddress>,
  1105. ll_dst_addr: Option<LlAddress>,
  1106. addr_context: &[AddressContext],
  1107. ) -> Result<Self> {
  1108. // Ensure basic accessors will work.
  1109. packet.check_len()?;
  1110. if packet.dispatch_field() != DISPATCH_IPHC_HEADER {
  1111. // This is not an LOWPAN_IPHC packet.
  1112. return Err(Error);
  1113. }
  1114. let src_addr = packet.src_addr()?.resolve(ll_src_addr, addr_context)?;
  1115. let dst_addr = packet.dst_addr()?.resolve(ll_dst_addr, addr_context)?;
  1116. Ok(Self {
  1117. src_addr,
  1118. ll_src_addr,
  1119. dst_addr,
  1120. ll_dst_addr,
  1121. next_header: packet.next_header(),
  1122. hop_limit: packet.hop_limit(),
  1123. ecn: packet.ecn_field(),
  1124. dscp: packet.dscp_field(),
  1125. flow_label: packet.flow_label_field(),
  1126. })
  1127. }
  1128. /// Return the length of a header that will be emitted from this high-level representation.
  1129. pub fn buffer_len(&self) -> usize {
  1130. let mut len = 0;
  1131. len += 2; // The minimal header length
  1132. len += match self.next_header {
  1133. NextHeader::Compressed => 0, // The next header is compressed (we don't need to inline what the next header is)
  1134. NextHeader::Uncompressed(_) => 1, // The next header field is inlined
  1135. };
  1136. // Hop Limit size
  1137. len += match self.hop_limit {
  1138. 255 | 64 | 1 => 0, // We can inline the hop limit
  1139. _ => 1,
  1140. };
  1141. // Add the length of the source address
  1142. len += if self.src_addr == ipv6::Address::UNSPECIFIED {
  1143. 0
  1144. } else if self.src_addr.is_link_local() {
  1145. let src = self.src_addr.as_bytes();
  1146. let ll = [src[14], src[15]];
  1147. let is_eui_64 = self
  1148. .ll_src_addr
  1149. .map(|addr| {
  1150. addr.as_eui_64()
  1151. .map(|addr| addr[..] == src[8..])
  1152. .unwrap_or(false)
  1153. })
  1154. .unwrap_or(false);
  1155. if src[8..14] == [0, 0, 0, 0xff, 0xfe, 0] {
  1156. if self.ll_src_addr == Some(LlAddress::Short(ll)) {
  1157. 0
  1158. } else {
  1159. 2
  1160. }
  1161. } else if is_eui_64 {
  1162. 0
  1163. } else {
  1164. 8
  1165. }
  1166. } else {
  1167. 16
  1168. };
  1169. // Add the size of the destination header
  1170. let dst = self.dst_addr.as_bytes();
  1171. len += if self.dst_addr.is_multicast() {
  1172. if dst[1] == 0x02 && dst[2..15] == [0; 13] {
  1173. 1
  1174. } else if dst[2..13] == [0; 11] {
  1175. 4
  1176. } else if dst[2..11] == [0; 9] {
  1177. 6
  1178. } else {
  1179. 16
  1180. }
  1181. } else if self.dst_addr.is_link_local() {
  1182. let is_eui_64 = self
  1183. .ll_dst_addr
  1184. .map(|addr| {
  1185. addr.as_eui_64()
  1186. .map(|addr| addr[..] == dst[8..])
  1187. .unwrap_or(false)
  1188. })
  1189. .unwrap_or(false);
  1190. if dst[8..14] == [0, 0, 0, 0xff, 0xfe, 0] {
  1191. let ll = [dst[14], dst[15]];
  1192. if self.ll_dst_addr == Some(LlAddress::Short(ll)) {
  1193. 0
  1194. } else {
  1195. 2
  1196. }
  1197. } else if is_eui_64 {
  1198. 0
  1199. } else {
  1200. 8
  1201. }
  1202. } else {
  1203. 16
  1204. };
  1205. len += match (self.ecn, self.dscp, self.flow_label) {
  1206. (Some(_), Some(_), Some(_)) => 4,
  1207. (Some(_), None, Some(_)) => 3,
  1208. (Some(_), Some(_), None) => 1,
  1209. (None, None, None) => 0,
  1210. _ => unreachable!(),
  1211. };
  1212. len
  1213. }
  1214. /// Emit a high-level representation into a 6LoWPAN IPHC header.
  1215. pub fn emit<T: AsRef<[u8]> + AsMut<[u8]>>(&self, packet: &mut Packet<T>) {
  1216. let idx = 2;
  1217. packet.set_dispatch_field();
  1218. // FIXME(thvdveld): we don't set anything from the traffic flow.
  1219. packet.set_tf_field(0b11);
  1220. let idx = packet.set_next_header(self.next_header, idx);
  1221. let idx = packet.set_hop_limit(self.hop_limit, idx);
  1222. let idx = packet.set_src_address(self.src_addr, self.ll_src_addr, idx);
  1223. packet.set_dst_address(self.dst_addr, self.ll_dst_addr, idx);
  1224. }
  1225. }
  1226. #[cfg(test)]
  1227. mod test {
  1228. use super::*;
  1229. #[test]
  1230. fn iphc_fields() {
  1231. let bytes = [
  1232. 0x7a, 0x33, // IPHC
  1233. 0x3a, // Next header
  1234. ];
  1235. let packet = Packet::new_unchecked(bytes);
  1236. assert_eq!(packet.dispatch_field(), 0b011);
  1237. assert_eq!(packet.tf_field(), 0b11);
  1238. assert_eq!(packet.nh_field(), 0b0);
  1239. assert_eq!(packet.hlim_field(), 0b10);
  1240. assert_eq!(packet.cid_field(), 0b0);
  1241. assert_eq!(packet.sac_field(), 0b0);
  1242. assert_eq!(packet.sam_field(), 0b11);
  1243. assert_eq!(packet.m_field(), 0b0);
  1244. assert_eq!(packet.dac_field(), 0b0);
  1245. assert_eq!(packet.dam_field(), 0b11);
  1246. assert_eq!(
  1247. packet.next_header(),
  1248. NextHeader::Uncompressed(IpProtocol::Icmpv6)
  1249. );
  1250. assert_eq!(packet.src_address_size(), 0);
  1251. assert_eq!(packet.dst_address_size(), 0);
  1252. assert_eq!(packet.hop_limit(), 64);
  1253. assert_eq!(
  1254. packet.src_addr(),
  1255. Ok(UnresolvedAddress::WithoutContext(AddressMode::FullyElided))
  1256. );
  1257. assert_eq!(
  1258. packet.dst_addr(),
  1259. Ok(UnresolvedAddress::WithoutContext(AddressMode::FullyElided))
  1260. );
  1261. let bytes = [
  1262. 0x7e, 0xf7, // IPHC,
  1263. 0x00, // CID
  1264. ];
  1265. let packet = Packet::new_unchecked(bytes);
  1266. assert_eq!(packet.dispatch_field(), 0b011);
  1267. assert_eq!(packet.tf_field(), 0b11);
  1268. assert_eq!(packet.nh_field(), 0b1);
  1269. assert_eq!(packet.hlim_field(), 0b10);
  1270. assert_eq!(packet.cid_field(), 0b1);
  1271. assert_eq!(packet.sac_field(), 0b1);
  1272. assert_eq!(packet.sam_field(), 0b11);
  1273. assert_eq!(packet.m_field(), 0b0);
  1274. assert_eq!(packet.dac_field(), 0b1);
  1275. assert_eq!(packet.dam_field(), 0b11);
  1276. assert_eq!(packet.next_header(), NextHeader::Compressed);
  1277. assert_eq!(packet.src_address_size(), 0);
  1278. assert_eq!(packet.dst_address_size(), 0);
  1279. assert_eq!(packet.hop_limit(), 64);
  1280. assert_eq!(
  1281. packet.src_addr(),
  1282. Ok(UnresolvedAddress::WithContext((
  1283. 0,
  1284. AddressMode::FullyElided
  1285. )))
  1286. );
  1287. assert_eq!(
  1288. packet.dst_addr(),
  1289. Ok(UnresolvedAddress::WithContext((
  1290. 0,
  1291. AddressMode::FullyElided
  1292. )))
  1293. );
  1294. }
  1295. }
  1296. }
  1297. pub mod nhc {
  1298. //! Implementation of Next Header Compression from [RFC 6282 § 4].
  1299. //!
  1300. //! [RFC 6282 § 4]: https://datatracker.ietf.org/doc/html/rfc6282#section-4
  1301. use super::{Error, NextHeader, Result, DISPATCH_EXT_HEADER, DISPATCH_UDP_HEADER};
  1302. use crate::{
  1303. phy::ChecksumCapabilities,
  1304. wire::{
  1305. ip::{checksum, Address as IpAddress},
  1306. ipv6,
  1307. udp::Repr as UdpRepr,
  1308. IpProtocol,
  1309. },
  1310. };
  1311. use byteorder::{ByteOrder, NetworkEndian};
  1312. use ipv6::Address;
  1313. macro_rules! get_field {
  1314. ($name:ident, $mask:expr, $shift:expr) => {
  1315. fn $name(&self) -> u8 {
  1316. let data = self.buffer.as_ref();
  1317. let raw = &data[0];
  1318. ((raw >> $shift) & $mask) as u8
  1319. }
  1320. };
  1321. }
  1322. macro_rules! set_field {
  1323. ($name:ident, $mask:expr, $shift:expr) => {
  1324. fn $name(&mut self, val: u8) {
  1325. let data = self.buffer.as_mut();
  1326. let mut raw = data[0];
  1327. raw = (raw & !($mask << $shift)) | (val << $shift);
  1328. data[0] = raw;
  1329. }
  1330. };
  1331. }
  1332. #[derive(Debug, Clone)]
  1333. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  1334. /// A read/write wrapper around a 6LoWPAN_NHC Header.
  1335. /// [RFC 6282 § 4.2] specifies the format of the header.
  1336. ///
  1337. /// The header has the following format:
  1338. /// ```txt
  1339. /// 0 1 2 3 4 5 6 7
  1340. /// +---+---+---+---+---+---+---+---+
  1341. /// | 1 | 1 | 1 | 0 | EID |NH |
  1342. /// +---+---+---+---+---+---+---+---+
  1343. /// ```
  1344. ///
  1345. /// With:
  1346. /// - EID: the extension header ID
  1347. /// - NH: Next Header
  1348. ///
  1349. /// [RFC 6282 § 4.2]: https://datatracker.ietf.org/doc/html/rfc6282#section-4.2
  1350. pub enum NhcPacket {
  1351. ExtHeader,
  1352. UdpHeader,
  1353. }
  1354. impl NhcPacket {
  1355. /// Returns the type of the Next Header header.
  1356. /// This can either be an Extenstion header or an 6LoWPAN Udp header.
  1357. ///
  1358. /// # Errors
  1359. /// Returns `[Error::Unrecognized]` when neither the Extension Header dispatch or the Udp
  1360. /// dispatch is recognized.
  1361. pub fn dispatch(buffer: impl AsRef<[u8]>) -> Result<Self> {
  1362. let raw = buffer.as_ref();
  1363. if raw.is_empty() {
  1364. return Err(Error);
  1365. }
  1366. if raw[0] >> 4 == DISPATCH_EXT_HEADER {
  1367. // We have a compressed IPv6 Extension Header.
  1368. Ok(Self::ExtHeader)
  1369. } else if raw[0] >> 3 == DISPATCH_UDP_HEADER {
  1370. // We have a compressed UDP header.
  1371. Ok(Self::UdpHeader)
  1372. } else {
  1373. Err(Error)
  1374. }
  1375. }
  1376. }
  1377. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  1378. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  1379. pub enum ExtHeaderId {
  1380. HopByHopHeader,
  1381. RoutingHeader,
  1382. FragmentHeader,
  1383. DestinationOptionsHeader,
  1384. MobilityHeader,
  1385. Header,
  1386. Reserved,
  1387. }
  1388. impl From<ExtHeaderId> for IpProtocol {
  1389. fn from(val: ExtHeaderId) -> Self {
  1390. match val {
  1391. ExtHeaderId::HopByHopHeader => Self::HopByHop,
  1392. ExtHeaderId::RoutingHeader => Self::Ipv6Route,
  1393. ExtHeaderId::FragmentHeader => Self::Ipv6Frag,
  1394. ExtHeaderId::DestinationOptionsHeader => Self::Ipv6Opts,
  1395. ExtHeaderId::MobilityHeader => Self::Unknown(0),
  1396. ExtHeaderId::Header => Self::Unknown(0),
  1397. ExtHeaderId::Reserved => Self::Unknown(0),
  1398. }
  1399. }
  1400. }
  1401. /// A read/write wrapper around a 6LoWPAN NHC Extension header.
  1402. #[derive(Debug, Clone)]
  1403. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  1404. pub struct ExtHeaderPacket<T: AsRef<[u8]>> {
  1405. buffer: T,
  1406. }
  1407. impl<T: AsRef<[u8]>> ExtHeaderPacket<T> {
  1408. /// Input a raw octet buffer with a 6LoWPAN NHC Extension Header structure.
  1409. pub const fn new_unchecked(buffer: T) -> Self {
  1410. ExtHeaderPacket { buffer }
  1411. }
  1412. /// Shorthand for a combination of [new_unchecked] and [check_len].
  1413. ///
  1414. /// [new_unchecked]: #method.new_unchecked
  1415. /// [check_len]: #method.check_len
  1416. pub fn new_checked(buffer: T) -> Result<Self> {
  1417. let packet = Self::new_unchecked(buffer);
  1418. packet.check_len()?;
  1419. if packet.eid_field() > 7 {
  1420. return Err(Error);
  1421. }
  1422. Ok(packet)
  1423. }
  1424. /// Ensure that no accessor method will panic if called.
  1425. /// Returns `Err(Error)` if the buffer is too short.
  1426. pub fn check_len(&self) -> Result<()> {
  1427. let buffer = self.buffer.as_ref();
  1428. if buffer.is_empty() {
  1429. return Err(Error);
  1430. }
  1431. let mut len = 1;
  1432. len += self.next_header_size();
  1433. if len <= buffer.len() {
  1434. Ok(())
  1435. } else {
  1436. Err(Error)
  1437. }
  1438. }
  1439. /// Consumes the frame, returning the underlying buffer.
  1440. pub fn into_inner(self) -> T {
  1441. self.buffer
  1442. }
  1443. get_field!(dispatch_field, 0b1111, 4);
  1444. get_field!(eid_field, 0b111, 1);
  1445. get_field!(nh_field, 0b1, 0);
  1446. /// Return the Extension Header ID.
  1447. pub fn extension_header_id(&self) -> ExtHeaderId {
  1448. match self.eid_field() {
  1449. 0 => ExtHeaderId::HopByHopHeader,
  1450. 1 => ExtHeaderId::RoutingHeader,
  1451. 2 => ExtHeaderId::FragmentHeader,
  1452. 3 => ExtHeaderId::DestinationOptionsHeader,
  1453. 4 => ExtHeaderId::MobilityHeader,
  1454. 5 | 6 => ExtHeaderId::Reserved,
  1455. 7 => ExtHeaderId::Header,
  1456. _ => unreachable!(),
  1457. }
  1458. }
  1459. /// Parse the next header field.
  1460. pub fn next_header(&self) -> NextHeader {
  1461. if self.nh_field() == 1 {
  1462. NextHeader::Compressed
  1463. } else {
  1464. // The full 8 bits for Next Header are carried in-line.
  1465. let start = 1;
  1466. let data = self.buffer.as_ref();
  1467. let nh = data[start];
  1468. NextHeader::Uncompressed(IpProtocol::from(nh))
  1469. }
  1470. }
  1471. /// Return the size of the Next Header field.
  1472. fn next_header_size(&self) -> usize {
  1473. // If nh is set, then the Next Header is compressed using LOWPAN_NHC
  1474. match self.nh_field() {
  1475. 0 => 1,
  1476. 1 => 0,
  1477. _ => unreachable!(),
  1478. }
  1479. }
  1480. }
  1481. impl<'a, T: AsRef<[u8]> + ?Sized> ExtHeaderPacket<&'a T> {
  1482. /// Return a pointer to the payload.
  1483. pub fn payload(&self) -> &'a [u8] {
  1484. let start = 2 + self.next_header_size();
  1485. &self.buffer.as_ref()[start..]
  1486. }
  1487. }
  1488. impl<T: AsRef<[u8]> + AsMut<[u8]>> ExtHeaderPacket<T> {
  1489. /// Return a mutable pointer to the payload.
  1490. pub fn payload_mut(&mut self) -> &mut [u8] {
  1491. let start = 2 + self.next_header_size();
  1492. &mut self.buffer.as_mut()[start..]
  1493. }
  1494. /// Set the dispatch field to `0b1110`.
  1495. fn set_dispatch_field(&mut self) {
  1496. let data = self.buffer.as_mut();
  1497. data[0] = (data[0] & !(0b1111 << 4)) | (DISPATCH_EXT_HEADER << 4);
  1498. }
  1499. set_field!(set_eid_field, 0b111, 1);
  1500. set_field!(set_nh_field, 0b1, 0);
  1501. /// Set the Extension Header ID field.
  1502. fn set_extension_header_id(&mut self, ext_header_id: ExtHeaderId) {
  1503. let id = match ext_header_id {
  1504. ExtHeaderId::HopByHopHeader => 0,
  1505. ExtHeaderId::RoutingHeader => 1,
  1506. ExtHeaderId::FragmentHeader => 2,
  1507. ExtHeaderId::DestinationOptionsHeader => 3,
  1508. ExtHeaderId::MobilityHeader => 4,
  1509. ExtHeaderId::Reserved => 5,
  1510. ExtHeaderId::Header => 7,
  1511. };
  1512. self.set_eid_field(id);
  1513. }
  1514. /// Set the Next Header.
  1515. fn set_next_header(&mut self, next_header: NextHeader) {
  1516. match next_header {
  1517. NextHeader::Compressed => self.set_nh_field(0b1),
  1518. NextHeader::Uncompressed(nh) => {
  1519. self.set_nh_field(0b0);
  1520. let start = 1;
  1521. let data = self.buffer.as_mut();
  1522. data[start] = nh.into();
  1523. }
  1524. }
  1525. }
  1526. /// Set the length.
  1527. fn set_length(&mut self, length: u8) {
  1528. let start = 1 + self.next_header_size();
  1529. let data = self.buffer.as_mut();
  1530. data[start] = length;
  1531. }
  1532. }
  1533. /// A high-level representation of an 6LoWPAN NHC Extension header.
  1534. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  1535. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  1536. pub struct ExtHeaderRepr {
  1537. ext_header_id: ExtHeaderId,
  1538. next_header: NextHeader,
  1539. length: u8,
  1540. }
  1541. impl ExtHeaderRepr {
  1542. /// Parse a 6LoWPAN NHC Extension Header packet and return a high-level representation.
  1543. pub fn parse<T: AsRef<[u8]> + ?Sized>(packet: &ExtHeaderPacket<&T>) -> Result<Self> {
  1544. // Ensure basic accessors will work.
  1545. packet.check_len()?;
  1546. if packet.dispatch_field() != DISPATCH_EXT_HEADER {
  1547. return Err(Error);
  1548. }
  1549. Ok(Self {
  1550. ext_header_id: packet.extension_header_id(),
  1551. next_header: packet.next_header(),
  1552. length: packet.payload().len() as u8,
  1553. })
  1554. }
  1555. /// Return the length of a header that will be emitted from this high-level representation.
  1556. pub fn buffer_len(&self) -> usize {
  1557. let mut len = 1; // The minimal header size
  1558. if self.next_header != NextHeader::Compressed {
  1559. len += 1;
  1560. }
  1561. len += 1; // The length
  1562. len
  1563. }
  1564. /// Emit a high-level representaiton into a 6LoWPAN NHC Extension Header packet.
  1565. pub fn emit<T: AsRef<[u8]> + AsMut<[u8]>>(&self, packet: &mut ExtHeaderPacket<T>) {
  1566. packet.set_dispatch_field();
  1567. packet.set_extension_header_id(self.ext_header_id);
  1568. packet.set_next_header(self.next_header);
  1569. packet.set_length(self.length);
  1570. }
  1571. }
  1572. #[cfg(test)]
  1573. mod tests {
  1574. use super::*;
  1575. use crate::wire::{Ipv6RoutingHeader, Ipv6RoutingRepr};
  1576. #[cfg(feature = "proto-rpl")]
  1577. use crate::wire::{
  1578. Ipv6Option, Ipv6OptionRepr, Ipv6OptionsIterator, RplHopByHopRepr, RplInstanceId,
  1579. };
  1580. #[cfg(feature = "proto-rpl")]
  1581. const RPL_HOP_BY_HOP_PACKET: [u8; 9] =
  1582. [0xe0, 0x3a, 0x06, 0x63, 0x04, 0x00, 0x1e, 0x03, 0x00];
  1583. const ROUTING_SR_PACKET: [u8; 32] = [
  1584. 0xe3, 0x1e, 0x03, 0x03, 0x99, 0x30, 0x00, 0x00, 0x05, 0x00, 0x05, 0x00, 0x05, 0x00,
  1585. 0x05, 0x06, 0x00, 0x06, 0x00, 0x06, 0x00, 0x06, 0x02, 0x00, 0x02, 0x00, 0x02, 0x00,
  1586. 0x02, 0x00, 0x00, 0x00,
  1587. ];
  1588. #[test]
  1589. #[cfg(feature = "proto-rpl")]
  1590. fn test_rpl_hop_by_hop_option_deconstruct() {
  1591. let header = ExtHeaderPacket::new_checked(&RPL_HOP_BY_HOP_PACKET).unwrap();
  1592. assert_eq!(
  1593. header.next_header(),
  1594. NextHeader::Uncompressed(IpProtocol::Icmpv6)
  1595. );
  1596. assert_eq!(header.extension_header_id(), ExtHeaderId::HopByHopHeader);
  1597. let options = header.payload();
  1598. let mut options = Ipv6OptionsIterator::new(options);
  1599. let rpl_repr = options.next().unwrap();
  1600. let rpl_repr = rpl_repr.unwrap();
  1601. match rpl_repr {
  1602. Ipv6OptionRepr::Rpl(rpl) => {
  1603. assert_eq!(
  1604. rpl,
  1605. RplHopByHopRepr {
  1606. down: false,
  1607. rank_error: false,
  1608. forwarding_error: false,
  1609. instance_id: RplInstanceId::from(0x1e),
  1610. sender_rank: 0x0300,
  1611. }
  1612. );
  1613. }
  1614. _ => unreachable!(),
  1615. }
  1616. }
  1617. #[test]
  1618. #[cfg(feature = "proto-rpl")]
  1619. fn test_rpl_hop_by_hop_option_emit() {
  1620. let repr = Ipv6OptionRepr::Rpl(RplHopByHopRepr {
  1621. down: false,
  1622. rank_error: false,
  1623. forwarding_error: false,
  1624. instance_id: RplInstanceId::from(0x1e),
  1625. sender_rank: 0x0300,
  1626. });
  1627. let ext_hdr = ExtHeaderRepr {
  1628. ext_header_id: ExtHeaderId::HopByHopHeader,
  1629. next_header: NextHeader::Uncompressed(IpProtocol::Icmpv6),
  1630. length: repr.buffer_len() as u8,
  1631. };
  1632. let mut buffer = vec![0u8; ext_hdr.buffer_len() + repr.buffer_len()];
  1633. ext_hdr.emit(&mut ExtHeaderPacket::new_unchecked(
  1634. &mut buffer[..ext_hdr.buffer_len()],
  1635. ));
  1636. repr.emit(&mut Ipv6Option::new_unchecked(
  1637. &mut buffer[ext_hdr.buffer_len()..],
  1638. ));
  1639. assert_eq!(&buffer[..], RPL_HOP_BY_HOP_PACKET);
  1640. }
  1641. #[test]
  1642. fn test_source_routing_deconstruct() {
  1643. let header = ExtHeaderPacket::new_checked(&ROUTING_SR_PACKET).unwrap();
  1644. assert_eq!(header.next_header(), NextHeader::Compressed);
  1645. assert_eq!(header.extension_header_id(), ExtHeaderId::RoutingHeader);
  1646. let routing_hdr = Ipv6RoutingHeader::new_checked(header.payload()).unwrap();
  1647. let repr = Ipv6RoutingRepr::parse(&routing_hdr).unwrap();
  1648. assert_eq!(
  1649. repr,
  1650. Ipv6RoutingRepr::Rpl {
  1651. segments_left: 3,
  1652. cmpr_i: 9,
  1653. cmpr_e: 9,
  1654. pad: 3,
  1655. addresses: &[
  1656. 0x05, 0x00, 0x05, 0x00, 0x05, 0x00, 0x05, 0x06, 0x00, 0x06, 0x00, 0x06,
  1657. 0x00, 0x06, 0x02, 0x00, 0x02, 0x00, 0x02, 0x00, 0x02, 0x00, 0x00, 0x00
  1658. ],
  1659. }
  1660. );
  1661. }
  1662. #[test]
  1663. fn test_source_routing_emit() {
  1664. let routing_hdr = Ipv6RoutingRepr::Rpl {
  1665. segments_left: 3,
  1666. cmpr_i: 9,
  1667. cmpr_e: 9,
  1668. pad: 3,
  1669. addresses: &[
  1670. 0x05, 0x00, 0x05, 0x00, 0x05, 0x00, 0x05, 0x06, 0x00, 0x06, 0x00, 0x06, 0x00,
  1671. 0x06, 0x02, 0x00, 0x02, 0x00, 0x02, 0x00, 0x02, 0x00, 0x00, 0x00,
  1672. ],
  1673. };
  1674. let ext_hdr = ExtHeaderRepr {
  1675. ext_header_id: ExtHeaderId::RoutingHeader,
  1676. next_header: NextHeader::Compressed,
  1677. length: routing_hdr.buffer_len() as u8,
  1678. };
  1679. let mut buffer = vec![0u8; ext_hdr.buffer_len() + routing_hdr.buffer_len()];
  1680. ext_hdr.emit(&mut ExtHeaderPacket::new_unchecked(
  1681. &mut buffer[..ext_hdr.buffer_len()],
  1682. ));
  1683. routing_hdr.emit(&mut Ipv6RoutingHeader::new_unchecked(
  1684. &mut buffer[ext_hdr.buffer_len()..],
  1685. ));
  1686. assert_eq!(&buffer[..], ROUTING_SR_PACKET);
  1687. }
  1688. }
  1689. /// A read/write wrapper around a 6LoWPAN_NHC UDP frame.
  1690. /// [RFC 6282 § 4.3] specifies the format of the header.
  1691. ///
  1692. /// The base header has the following formath:
  1693. /// ```txt
  1694. /// 0 1 2 3 4 5 6 7
  1695. /// +---+---+---+---+---+---+---+---+
  1696. /// | 1 | 1 | 1 | 1 | 0 | C | P |
  1697. /// +---+---+---+---+---+---+---+---+
  1698. /// With:
  1699. /// - C: checksum, specifies if the checksum is elided.
  1700. /// - P: ports, specifies if the ports are elided.
  1701. /// ```
  1702. ///
  1703. /// [RFC 6282 § 4.3]: https://datatracker.ietf.org/doc/html/rfc6282#section-4.3
  1704. #[derive(Debug, Clone)]
  1705. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  1706. pub struct UdpNhcPacket<T: AsRef<[u8]>> {
  1707. buffer: T,
  1708. }
  1709. impl<T: AsRef<[u8]>> UdpNhcPacket<T> {
  1710. /// Input a raw octet buffer with a LOWPAN_NHC frame structure for UDP.
  1711. pub const fn new_unchecked(buffer: T) -> Self {
  1712. Self { buffer }
  1713. }
  1714. /// Shorthand for a combination of [new_unchecked] and [check_len].
  1715. ///
  1716. /// [new_unchecked]: #method.new_unchecked
  1717. /// [check_len]: #method.check_len
  1718. pub fn new_checked(buffer: T) -> Result<Self> {
  1719. let packet = Self::new_unchecked(buffer);
  1720. packet.check_len()?;
  1721. Ok(packet)
  1722. }
  1723. /// Ensure that no accessor method will panic if called.
  1724. /// Returns `Err(Error::Truncated)` if the buffer is too short.
  1725. pub fn check_len(&self) -> Result<()> {
  1726. let buffer = self.buffer.as_ref();
  1727. if buffer.is_empty() {
  1728. return Err(Error);
  1729. }
  1730. let index = 1 + self.ports_size() + self.checksum_size();
  1731. if index > buffer.len() {
  1732. return Err(Error);
  1733. }
  1734. Ok(())
  1735. }
  1736. /// Consumes the frame, returning the underlying buffer.
  1737. pub fn into_inner(self) -> T {
  1738. self.buffer
  1739. }
  1740. get_field!(dispatch_field, 0b11111, 3);
  1741. get_field!(checksum_field, 0b1, 2);
  1742. get_field!(ports_field, 0b11, 0);
  1743. /// Returns the index of the start of the next header compressed fields.
  1744. const fn nhc_fields_start(&self) -> usize {
  1745. 1
  1746. }
  1747. /// Return the source port number.
  1748. pub fn src_port(&self) -> u16 {
  1749. match self.ports_field() {
  1750. 0b00 | 0b01 => {
  1751. // The full 16 bits are carried in-line.
  1752. let data = self.buffer.as_ref();
  1753. let start = self.nhc_fields_start();
  1754. NetworkEndian::read_u16(&data[start..start + 2])
  1755. }
  1756. 0b10 => {
  1757. // The first 8 bits are elided.
  1758. let data = self.buffer.as_ref();
  1759. let start = self.nhc_fields_start();
  1760. 0xf000 + data[start] as u16
  1761. }
  1762. 0b11 => {
  1763. // The first 12 bits are elided.
  1764. let data = self.buffer.as_ref();
  1765. let start = self.nhc_fields_start();
  1766. 0xf0b0 + (data[start] >> 4) as u16
  1767. }
  1768. _ => unreachable!(),
  1769. }
  1770. }
  1771. /// Return the destination port number.
  1772. pub fn dst_port(&self) -> u16 {
  1773. match self.ports_field() {
  1774. 0b00 => {
  1775. // The full 16 bits are carried in-line.
  1776. let data = self.buffer.as_ref();
  1777. let idx = self.nhc_fields_start();
  1778. NetworkEndian::read_u16(&data[idx + 2..idx + 4])
  1779. }
  1780. 0b01 => {
  1781. // The first 8 bits are elided.
  1782. let data = self.buffer.as_ref();
  1783. let idx = self.nhc_fields_start();
  1784. 0xf000 + data[idx] as u16
  1785. }
  1786. 0b10 => {
  1787. // The full 16 bits are carried in-line.
  1788. let data = self.buffer.as_ref();
  1789. let idx = self.nhc_fields_start();
  1790. NetworkEndian::read_u16(&data[idx + 1..idx + 1 + 2])
  1791. }
  1792. 0b11 => {
  1793. // The first 12 bits are elided.
  1794. let data = self.buffer.as_ref();
  1795. let start = self.nhc_fields_start();
  1796. 0xf0b0 + (data[start] & 0xff) as u16
  1797. }
  1798. _ => unreachable!(),
  1799. }
  1800. }
  1801. /// Return the checksum.
  1802. pub fn checksum(&self) -> Option<u16> {
  1803. if self.checksum_field() == 0b0 {
  1804. // The first 12 bits are elided.
  1805. let data = self.buffer.as_ref();
  1806. let start = self.nhc_fields_start() + self.ports_size();
  1807. Some(NetworkEndian::read_u16(&data[start..start + 2]))
  1808. } else {
  1809. // The checksum is elided and needs to be recomputed on the 6LoWPAN termination point.
  1810. None
  1811. }
  1812. }
  1813. // Return the size of the checksum field.
  1814. pub(crate) fn checksum_size(&self) -> usize {
  1815. match self.checksum_field() {
  1816. 0b0 => 2,
  1817. 0b1 => 0,
  1818. _ => unreachable!(),
  1819. }
  1820. }
  1821. /// Returns the total size of both port numbers.
  1822. pub(crate) fn ports_size(&self) -> usize {
  1823. match self.ports_field() {
  1824. 0b00 => 4, // 16 bits + 16 bits
  1825. 0b01 => 3, // 16 bits + 8 bits
  1826. 0b10 => 3, // 8 bits + 16 bits
  1827. 0b11 => 1, // 4 bits + 4 bits
  1828. _ => unreachable!(),
  1829. }
  1830. }
  1831. }
  1832. impl<'a, T: AsRef<[u8]> + ?Sized> UdpNhcPacket<&'a T> {
  1833. /// Return a pointer to the payload.
  1834. pub fn payload(&self) -> &'a [u8] {
  1835. let start = 1 + self.ports_size() + self.checksum_size();
  1836. &self.buffer.as_ref()[start..]
  1837. }
  1838. }
  1839. impl<T: AsRef<[u8]> + AsMut<[u8]>> UdpNhcPacket<T> {
  1840. /// Return a mutable pointer to the payload.
  1841. pub fn payload_mut(&mut self) -> &mut [u8] {
  1842. let start = 1 + self.ports_size() + 2; // XXX(thvdveld): we assume we put the checksum inlined.
  1843. &mut self.buffer.as_mut()[start..]
  1844. }
  1845. /// Set the dispatch field to `0b11110`.
  1846. fn set_dispatch_field(&mut self) {
  1847. let data = self.buffer.as_mut();
  1848. data[0] = (data[0] & !(0b11111 << 3)) | (DISPATCH_UDP_HEADER << 3);
  1849. }
  1850. set_field!(set_checksum_field, 0b1, 2);
  1851. set_field!(set_ports_field, 0b11, 0);
  1852. fn set_ports(&mut self, src_port: u16, dst_port: u16) {
  1853. let mut idx = 1;
  1854. match (src_port, dst_port) {
  1855. (0xf0b0..=0xf0bf, 0xf0b0..=0xf0bf) => {
  1856. // We can compress both the source and destination ports.
  1857. self.set_ports_field(0b11);
  1858. let data = self.buffer.as_mut();
  1859. data[idx] = (((src_port - 0xf0b0) as u8) << 4) & ((dst_port - 0xf0b0) as u8);
  1860. }
  1861. (0xf000..=0xf0ff, _) => {
  1862. // We can compress the source port, but not the destination port.
  1863. self.set_ports_field(0b10);
  1864. let data = self.buffer.as_mut();
  1865. data[idx] = (src_port - 0xf000) as u8;
  1866. idx += 1;
  1867. NetworkEndian::write_u16(&mut data[idx..idx + 2], dst_port);
  1868. }
  1869. (_, 0xf000..=0xf0ff) => {
  1870. // We can compress the destination port, but not the source port.
  1871. self.set_ports_field(0b01);
  1872. let data = self.buffer.as_mut();
  1873. NetworkEndian::write_u16(&mut data[idx..idx + 2], src_port);
  1874. idx += 2;
  1875. data[idx] = (dst_port - 0xf000) as u8;
  1876. }
  1877. (_, _) => {
  1878. // We cannot compress any port.
  1879. self.set_ports_field(0b00);
  1880. let data = self.buffer.as_mut();
  1881. NetworkEndian::write_u16(&mut data[idx..idx + 2], src_port);
  1882. idx += 2;
  1883. NetworkEndian::write_u16(&mut data[idx..idx + 2], dst_port);
  1884. }
  1885. };
  1886. }
  1887. fn set_checksum(&mut self, checksum: u16) {
  1888. self.set_checksum_field(0b0);
  1889. let idx = 1 + self.ports_size();
  1890. let data = self.buffer.as_mut();
  1891. NetworkEndian::write_u16(&mut data[idx..idx + 2], checksum);
  1892. }
  1893. }
  1894. /// A high-level representation of a 6LoWPAN NHC UDP header.
  1895. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  1896. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  1897. pub struct UdpNhcRepr(pub UdpRepr);
  1898. impl<'a> UdpNhcRepr {
  1899. /// Parse a 6LoWPAN NHC UDP packet and return a high-level representation.
  1900. pub fn parse<T: AsRef<[u8]> + ?Sized>(
  1901. packet: &UdpNhcPacket<&'a T>,
  1902. src_addr: &ipv6::Address,
  1903. dst_addr: &ipv6::Address,
  1904. checksum_caps: &ChecksumCapabilities,
  1905. ) -> Result<Self> {
  1906. packet.check_len()?;
  1907. if packet.dispatch_field() != DISPATCH_UDP_HEADER {
  1908. return Err(Error);
  1909. }
  1910. if checksum_caps.udp.rx() {
  1911. let payload_len = packet.payload().len();
  1912. let chk_sum = !checksum::combine(&[
  1913. checksum::pseudo_header(
  1914. &IpAddress::Ipv6(*src_addr),
  1915. &IpAddress::Ipv6(*dst_addr),
  1916. crate::wire::ip::Protocol::Udp,
  1917. payload_len as u32 + 8,
  1918. ),
  1919. packet.src_port(),
  1920. packet.dst_port(),
  1921. payload_len as u16 + 8,
  1922. checksum::data(packet.payload()),
  1923. ]);
  1924. if let Some(checksum) = packet.checksum() {
  1925. if chk_sum != checksum {
  1926. return Err(Error);
  1927. }
  1928. }
  1929. }
  1930. Ok(Self(UdpRepr {
  1931. src_port: packet.src_port(),
  1932. dst_port: packet.dst_port(),
  1933. }))
  1934. }
  1935. /// Return the length of a packet that will be emitted from this high-level representation.
  1936. pub fn header_len(&self) -> usize {
  1937. let mut len = 1; // The minimal header size
  1938. len += 2; // XXX We assume we will add the checksum at the end
  1939. // Check if we can compress the source and destination ports
  1940. match (self.src_port, self.dst_port) {
  1941. (0xf0b0..=0xf0bf, 0xf0b0..=0xf0bf) => len + 1,
  1942. (0xf000..=0xf0ff, _) | (_, 0xf000..=0xf0ff) => len + 3,
  1943. (_, _) => len + 4,
  1944. }
  1945. }
  1946. /// Emit a high-level representation into a LOWPAN_NHC UDP header.
  1947. pub fn emit<T: AsRef<[u8]> + AsMut<[u8]>>(
  1948. &self,
  1949. packet: &mut UdpNhcPacket<T>,
  1950. src_addr: &Address,
  1951. dst_addr: &Address,
  1952. payload_len: usize,
  1953. emit_payload: impl FnOnce(&mut [u8]),
  1954. ) {
  1955. packet.set_dispatch_field();
  1956. packet.set_ports(self.src_port, self.dst_port);
  1957. emit_payload(packet.payload_mut());
  1958. let chk_sum = !checksum::combine(&[
  1959. checksum::pseudo_header(
  1960. &IpAddress::Ipv6(*src_addr),
  1961. &IpAddress::Ipv6(*dst_addr),
  1962. crate::wire::ip::Protocol::Udp,
  1963. payload_len as u32 + 8,
  1964. ),
  1965. self.src_port,
  1966. self.dst_port,
  1967. payload_len as u16 + 8,
  1968. checksum::data(packet.payload_mut()),
  1969. ]);
  1970. packet.set_checksum(chk_sum);
  1971. }
  1972. }
  1973. impl core::ops::Deref for UdpNhcRepr {
  1974. type Target = UdpRepr;
  1975. fn deref(&self) -> &Self::Target {
  1976. &self.0
  1977. }
  1978. }
  1979. impl core::ops::DerefMut for UdpNhcRepr {
  1980. fn deref_mut(&mut self) -> &mut Self::Target {
  1981. &mut self.0
  1982. }
  1983. }
  1984. #[cfg(test)]
  1985. mod test {
  1986. use super::*;
  1987. #[test]
  1988. fn ext_header_nhc_fields() {
  1989. let bytes = [0xe3, 0x06, 0x03, 0x00, 0xff, 0x00, 0x00, 0x00];
  1990. let packet = ExtHeaderPacket::new_checked(&bytes[..]).unwrap();
  1991. assert_eq!(packet.next_header_size(), 0);
  1992. assert_eq!(packet.dispatch_field(), DISPATCH_EXT_HEADER);
  1993. assert_eq!(packet.extension_header_id(), ExtHeaderId::RoutingHeader);
  1994. assert_eq!(packet.payload(), [0x03, 0x00, 0xff, 0x00, 0x00, 0x00]);
  1995. }
  1996. #[test]
  1997. fn ext_header_emit() {
  1998. let ext_header = ExtHeaderRepr {
  1999. ext_header_id: ExtHeaderId::RoutingHeader,
  2000. next_header: NextHeader::Compressed,
  2001. length: 6,
  2002. };
  2003. let len = ext_header.buffer_len();
  2004. let mut buffer = [0u8; 127];
  2005. let mut packet = ExtHeaderPacket::new_unchecked(&mut buffer[..len]);
  2006. ext_header.emit(&mut packet);
  2007. assert_eq!(packet.dispatch_field(), DISPATCH_EXT_HEADER);
  2008. assert_eq!(packet.next_header(), NextHeader::Compressed);
  2009. assert_eq!(packet.extension_header_id(), ExtHeaderId::RoutingHeader);
  2010. }
  2011. #[test]
  2012. fn udp_nhc_fields() {
  2013. let bytes = [0xf0, 0x16, 0x2e, 0x22, 0x3d, 0x28, 0xc4];
  2014. let packet = UdpNhcPacket::new_checked(&bytes[..]).unwrap();
  2015. assert_eq!(packet.dispatch_field(), DISPATCH_UDP_HEADER);
  2016. assert_eq!(packet.checksum(), Some(0x28c4));
  2017. assert_eq!(packet.src_port(), 5678);
  2018. assert_eq!(packet.dst_port(), 8765);
  2019. }
  2020. #[test]
  2021. fn udp_emit() {
  2022. let udp = UdpNhcRepr(UdpRepr {
  2023. src_port: 0xf0b1,
  2024. dst_port: 0xf001,
  2025. });
  2026. let payload = b"Hello World!";
  2027. let src_addr = ipv6::Address::default();
  2028. let dst_addr = ipv6::Address::default();
  2029. let len = udp.header_len() + payload.len();
  2030. let mut buffer = [0u8; 127];
  2031. let mut packet = UdpNhcPacket::new_unchecked(&mut buffer[..len]);
  2032. udp.emit(&mut packet, &src_addr, &dst_addr, payload.len(), |buf| {
  2033. buf.copy_from_slice(&payload[..])
  2034. });
  2035. assert_eq!(packet.dispatch_field(), DISPATCH_UDP_HEADER);
  2036. assert_eq!(packet.src_port(), 0xf0b1);
  2037. assert_eq!(packet.dst_port(), 0xf001);
  2038. assert_eq!(packet.payload_mut(), b"Hello World!");
  2039. }
  2040. }
  2041. }
  2042. #[cfg(test)]
  2043. mod test {
  2044. use super::*;
  2045. #[test]
  2046. fn sixlowpan_fragment_emit() {
  2047. let repr = frag::Repr::FirstFragment {
  2048. size: 0xff,
  2049. tag: 0xabcd,
  2050. };
  2051. let buffer = [0u8; 4];
  2052. let mut packet = frag::Packet::new_unchecked(buffer);
  2053. assert_eq!(repr.buffer_len(), 4);
  2054. repr.emit(&mut packet);
  2055. assert_eq!(packet.datagram_size(), 0xff);
  2056. assert_eq!(packet.datagram_tag(), 0xabcd);
  2057. assert_eq!(packet.into_inner(), [0xc0, 0xff, 0xab, 0xcd]);
  2058. let repr = frag::Repr::Fragment {
  2059. size: 0xff,
  2060. tag: 0xabcd,
  2061. offset: 0xcc,
  2062. };
  2063. let buffer = [0u8; 5];
  2064. let mut packet = frag::Packet::new_unchecked(buffer);
  2065. assert_eq!(repr.buffer_len(), 5);
  2066. repr.emit(&mut packet);
  2067. assert_eq!(packet.datagram_size(), 0xff);
  2068. assert_eq!(packet.datagram_tag(), 0xabcd);
  2069. assert_eq!(packet.into_inner(), [0xe0, 0xff, 0xab, 0xcd, 0xcc]);
  2070. }
  2071. #[test]
  2072. fn sixlowpan_three_fragments() {
  2073. use crate::wire::ieee802154::Frame as Ieee802154Frame;
  2074. use crate::wire::ieee802154::Repr as Ieee802154Repr;
  2075. use crate::wire::Ieee802154Address;
  2076. let key = frag::Key {
  2077. ll_src_addr: Ieee802154Address::Extended([50, 147, 130, 47, 40, 8, 62, 217]),
  2078. ll_dst_addr: Ieee802154Address::Extended([26, 11, 66, 66, 66, 66, 66, 66]),
  2079. datagram_size: 307,
  2080. datagram_tag: 63,
  2081. };
  2082. let frame1: &[u8] = &[
  2083. 0x41, 0xcc, 0x92, 0xef, 0xbe, 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x0b, 0x1a, 0xd9,
  2084. 0x3e, 0x08, 0x28, 0x2f, 0x82, 0x93, 0x32, 0xc1, 0x33, 0x00, 0x3f, 0x6e, 0x33, 0x02,
  2085. 0x35, 0x3d, 0xf0, 0xd2, 0x5f, 0x1b, 0x39, 0xb4, 0x6b, 0x4c, 0x6f, 0x72, 0x65, 0x6d,
  2086. 0x20, 0x69, 0x70, 0x73, 0x75, 0x6d, 0x20, 0x64, 0x6f, 0x6c, 0x6f, 0x72, 0x20, 0x73,
  2087. 0x69, 0x74, 0x20, 0x61, 0x6d, 0x65, 0x74, 0x2c, 0x20, 0x63, 0x6f, 0x6e, 0x73, 0x65,
  2088. 0x63, 0x74, 0x65, 0x74, 0x75, 0x72, 0x20, 0x61, 0x64, 0x69, 0x70, 0x69, 0x73, 0x63,
  2089. 0x69, 0x6e, 0x67, 0x20, 0x65, 0x6c, 0x69, 0x74, 0x2e, 0x20, 0x41, 0x6c, 0x69, 0x71,
  2090. 0x75, 0x61, 0x6d, 0x20, 0x64, 0x75, 0x69, 0x20, 0x6f, 0x64, 0x69, 0x6f, 0x2c, 0x20,
  2091. 0x69, 0x61, 0x63, 0x75, 0x6c, 0x69, 0x73, 0x20, 0x76, 0x65, 0x6c, 0x20, 0x72,
  2092. ];
  2093. let ieee802154_frame = Ieee802154Frame::new_checked(frame1).unwrap();
  2094. let ieee802154_repr = Ieee802154Repr::parse(&ieee802154_frame).unwrap();
  2095. let sixlowpan_frame =
  2096. SixlowpanPacket::dispatch(ieee802154_frame.payload().unwrap()).unwrap();
  2097. let frag = if let SixlowpanPacket::FragmentHeader = sixlowpan_frame {
  2098. frag::Packet::new_checked(ieee802154_frame.payload().unwrap()).unwrap()
  2099. } else {
  2100. unreachable!()
  2101. };
  2102. assert_eq!(frag.datagram_size(), 307);
  2103. assert_eq!(frag.datagram_tag(), 0x003f);
  2104. assert_eq!(frag.datagram_offset(), 0);
  2105. assert_eq!(frag.get_key(&ieee802154_repr), key);
  2106. let frame2: &[u8] = &[
  2107. 0x41, 0xcc, 0x93, 0xef, 0xbe, 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x0b, 0x1a, 0xd9,
  2108. 0x3e, 0x08, 0x28, 0x2f, 0x82, 0x93, 0x32, 0xe1, 0x33, 0x00, 0x3f, 0x11, 0x75, 0x74,
  2109. 0x72, 0x75, 0x6d, 0x20, 0x61, 0x74, 0x2c, 0x20, 0x74, 0x72, 0x69, 0x73, 0x74, 0x69,
  2110. 0x71, 0x75, 0x65, 0x20, 0x6e, 0x6f, 0x6e, 0x20, 0x6e, 0x75, 0x6e, 0x63, 0x20, 0x65,
  2111. 0x72, 0x61, 0x74, 0x20, 0x63, 0x75, 0x72, 0x61, 0x65, 0x2e, 0x20, 0x4c, 0x6f, 0x72,
  2112. 0x65, 0x6d, 0x20, 0x69, 0x70, 0x73, 0x75, 0x6d, 0x20, 0x64, 0x6f, 0x6c, 0x6f, 0x72,
  2113. 0x20, 0x73, 0x69, 0x74, 0x20, 0x61, 0x6d, 0x65, 0x74, 0x2c, 0x20, 0x63, 0x6f, 0x6e,
  2114. 0x73, 0x65, 0x63, 0x74, 0x65, 0x74, 0x75, 0x72, 0x20, 0x61, 0x64, 0x69, 0x70, 0x69,
  2115. 0x73, 0x63, 0x69, 0x6e, 0x67, 0x20, 0x65, 0x6c, 0x69, 0x74,
  2116. ];
  2117. let ieee802154_frame = Ieee802154Frame::new_checked(frame2).unwrap();
  2118. let ieee802154_repr = Ieee802154Repr::parse(&ieee802154_frame).unwrap();
  2119. let sixlowpan_frame =
  2120. SixlowpanPacket::dispatch(ieee802154_frame.payload().unwrap()).unwrap();
  2121. let frag = if let SixlowpanPacket::FragmentHeader = sixlowpan_frame {
  2122. frag::Packet::new_checked(ieee802154_frame.payload().unwrap()).unwrap()
  2123. } else {
  2124. unreachable!()
  2125. };
  2126. assert_eq!(frag.datagram_size(), 307);
  2127. assert_eq!(frag.datagram_tag(), 0x003f);
  2128. assert_eq!(frag.datagram_offset(), 136 / 8);
  2129. assert_eq!(frag.get_key(&ieee802154_repr), key);
  2130. let frame3: &[u8] = &[
  2131. 0x41, 0xcc, 0x94, 0xef, 0xbe, 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x0b, 0x1a, 0xd9,
  2132. 0x3e, 0x08, 0x28, 0x2f, 0x82, 0x93, 0x32, 0xe1, 0x33, 0x00, 0x3f, 0x1d, 0x2e, 0x20,
  2133. 0x41, 0x6c, 0x69, 0x71, 0x75, 0x61, 0x6d, 0x20, 0x64, 0x75, 0x69, 0x20, 0x6f, 0x64,
  2134. 0x69, 0x6f, 0x2c, 0x20, 0x69, 0x61, 0x63, 0x75, 0x6c, 0x69, 0x73, 0x20, 0x76, 0x65,
  2135. 0x6c, 0x20, 0x72, 0x75, 0x74, 0x72, 0x75, 0x6d, 0x20, 0x61, 0x74, 0x2c, 0x20, 0x74,
  2136. 0x72, 0x69, 0x73, 0x74, 0x69, 0x71, 0x75, 0x65, 0x20, 0x6e, 0x6f, 0x6e, 0x20, 0x6e,
  2137. 0x75, 0x6e, 0x63, 0x20, 0x65, 0x72, 0x61, 0x74, 0x20, 0x63, 0x75, 0x72, 0x61, 0x65,
  2138. 0x2e, 0x20, 0x0a,
  2139. ];
  2140. let ieee802154_frame = Ieee802154Frame::new_checked(frame3).unwrap();
  2141. let ieee802154_repr = Ieee802154Repr::parse(&ieee802154_frame).unwrap();
  2142. let sixlowpan_frame =
  2143. SixlowpanPacket::dispatch(ieee802154_frame.payload().unwrap()).unwrap();
  2144. let frag = if let SixlowpanPacket::FragmentHeader = sixlowpan_frame {
  2145. frag::Packet::new_checked(ieee802154_frame.payload().unwrap()).unwrap()
  2146. } else {
  2147. unreachable!()
  2148. };
  2149. assert_eq!(frag.datagram_size(), 307);
  2150. assert_eq!(frag.datagram_tag(), 0x003f);
  2151. assert_eq!(frag.datagram_offset(), 232 / 8);
  2152. assert_eq!(frag.get_key(&ieee802154_repr), key);
  2153. }
  2154. }