ipv6.rs 42 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287
  1. #![deny(missing_docs)]
  2. use byteorder::{ByteOrder, NetworkEndian};
  3. use core::fmt;
  4. use super::{Error, Result};
  5. use crate::wire::ip::pretty_print_ip_payload;
  6. #[cfg(feature = "proto-ipv4")]
  7. use crate::wire::ipv4;
  8. pub use super::IpProtocol as Protocol;
  9. /// Minimum MTU required of all links supporting IPv6. See [RFC 8200 § 5].
  10. ///
  11. /// [RFC 8200 § 5]: https://tools.ietf.org/html/rfc8200#section-5
  12. pub const MIN_MTU: usize = 1280;
  13. /// Size of IPv6 adderess in octets.
  14. ///
  15. /// [RFC 8200 § 2]: https://www.rfc-editor.org/rfc/rfc4291#section-2
  16. pub const ADDR_SIZE: usize = 16;
  17. /// Size of IPv4-mapping prefix in octets.
  18. ///
  19. /// [RFC 8200 § 2]: https://www.rfc-editor.org/rfc/rfc4291#section-2
  20. pub const IPV4_MAPPED_PREFIX_SIZE: usize = ADDR_SIZE - 4; // 4 == ipv4::ADDR_SIZE , cannot DRY here because of dependency on a IPv4 module which is behind the feature
  21. /// A sixteen-octet IPv6 address.
  22. #[derive(Debug, Hash, PartialEq, Eq, PartialOrd, Ord, Clone, Copy, Default)]
  23. pub struct Address(pub [u8; ADDR_SIZE]);
  24. impl Address {
  25. /// The [unspecified address].
  26. ///
  27. /// [unspecified address]: https://tools.ietf.org/html/rfc4291#section-2.5.2
  28. pub const UNSPECIFIED: Address = Address([0x00; ADDR_SIZE]);
  29. /// The link-local [all nodes multicast address].
  30. ///
  31. /// [all nodes multicast address]: https://tools.ietf.org/html/rfc4291#section-2.7.1
  32. pub const LINK_LOCAL_ALL_NODES: Address = Address([
  33. 0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  34. 0x01,
  35. ]);
  36. /// The link-local [all routers multicast address].
  37. ///
  38. /// [all routers multicast address]: https://tools.ietf.org/html/rfc4291#section-2.7.1
  39. pub const LINK_LOCAL_ALL_ROUTERS: Address = Address([
  40. 0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  41. 0x02,
  42. ]);
  43. /// The [loopback address].
  44. ///
  45. /// [loopback address]: https://tools.ietf.org/html/rfc4291#section-2.5.3
  46. pub const LOOPBACK: Address = Address([
  47. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  48. 0x01,
  49. ]);
  50. /// The prefix used in [IPv4-mapped addresses].
  51. ///
  52. /// [IPv4-mapped addresses]: https://www.rfc-editor.org/rfc/rfc4291#section-2.5.5.2
  53. pub const IPV4_MAPPED_PREFIX: [u8; IPV4_MAPPED_PREFIX_SIZE] =
  54. [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff];
  55. /// Construct an IPv6 address from parts.
  56. #[allow(clippy::too_many_arguments)]
  57. pub const fn new(
  58. a0: u16,
  59. a1: u16,
  60. a2: u16,
  61. a3: u16,
  62. a4: u16,
  63. a5: u16,
  64. a6: u16,
  65. a7: u16,
  66. ) -> Address {
  67. Address([
  68. (a0 >> 8) as u8,
  69. a0 as u8,
  70. (a1 >> 8) as u8,
  71. a1 as u8,
  72. (a2 >> 8) as u8,
  73. a2 as u8,
  74. (a3 >> 8) as u8,
  75. a3 as u8,
  76. (a4 >> 8) as u8,
  77. a4 as u8,
  78. (a5 >> 8) as u8,
  79. a5 as u8,
  80. (a6 >> 8) as u8,
  81. a6 as u8,
  82. (a7 >> 8) as u8,
  83. a7 as u8,
  84. ])
  85. }
  86. /// Construct an IPv6 address from a sequence of octets, in big-endian.
  87. ///
  88. /// # Panics
  89. /// The function panics if `data` is not sixteen octets long.
  90. pub fn from_bytes(data: &[u8]) -> Address {
  91. let mut bytes = [0; ADDR_SIZE];
  92. bytes.copy_from_slice(data);
  93. Address(bytes)
  94. }
  95. /// Construct an IPv6 address from a sequence of words, in big-endian.
  96. ///
  97. /// # Panics
  98. /// The function panics if `data` is not 8 words long.
  99. pub fn from_parts(data: &[u16]) -> Address {
  100. assert!(data.len() >= 8);
  101. let mut bytes = [0; ADDR_SIZE];
  102. for (word_idx, chunk) in bytes.chunks_mut(2).enumerate() {
  103. NetworkEndian::write_u16(chunk, data[word_idx]);
  104. }
  105. Address(bytes)
  106. }
  107. /// Write a IPv6 address to the given slice.
  108. ///
  109. /// # Panics
  110. /// The function panics if `data` is not 8 words long.
  111. pub fn write_parts(&self, data: &mut [u16]) {
  112. assert!(data.len() >= 8);
  113. for (i, chunk) in self.0.chunks(2).enumerate() {
  114. data[i] = NetworkEndian::read_u16(chunk);
  115. }
  116. }
  117. /// Return an IPv6 address as a sequence of octets, in big-endian.
  118. pub const fn as_bytes(&self) -> &[u8] {
  119. &self.0
  120. }
  121. /// Query whether the IPv6 address is an [unicast address].
  122. ///
  123. /// [unicast address]: https://tools.ietf.org/html/rfc4291#section-2.5
  124. pub fn is_unicast(&self) -> bool {
  125. !(self.is_multicast() || self.is_unspecified())
  126. }
  127. /// Query whether the IPv6 address is a [multicast address].
  128. ///
  129. /// [multicast address]: https://tools.ietf.org/html/rfc4291#section-2.7
  130. pub const fn is_multicast(&self) -> bool {
  131. self.0[0] == 0xff
  132. }
  133. /// Query whether the IPv6 address is the [unspecified address].
  134. ///
  135. /// [unspecified address]: https://tools.ietf.org/html/rfc4291#section-2.5.2
  136. pub fn is_unspecified(&self) -> bool {
  137. self.0 == [0x00; ADDR_SIZE]
  138. }
  139. /// Query whether the IPv6 address is in the [link-local] scope.
  140. ///
  141. /// [link-local]: https://tools.ietf.org/html/rfc4291#section-2.5.6
  142. pub fn is_link_local(&self) -> bool {
  143. self.0[0..8] == [0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00]
  144. }
  145. /// Query whether the IPv6 address is the [loopback address].
  146. ///
  147. /// [loopback address]: https://tools.ietf.org/html/rfc4291#section-2.5.3
  148. pub fn is_loopback(&self) -> bool {
  149. *self == Self::LOOPBACK
  150. }
  151. /// Query whether the IPv6 address is an [IPv4 mapped IPv6 address].
  152. ///
  153. /// [IPv4 mapped IPv6 address]: https://tools.ietf.org/html/rfc4291#section-2.5.5.2
  154. pub fn is_ipv4_mapped(&self) -> bool {
  155. self.0[..IPV4_MAPPED_PREFIX_SIZE] == Self::IPV4_MAPPED_PREFIX
  156. }
  157. #[cfg(feature = "proto-ipv4")]
  158. /// Convert an IPv4 mapped IPv6 address to an IPv4 address.
  159. pub fn as_ipv4(&self) -> Option<ipv4::Address> {
  160. if self.is_ipv4_mapped() {
  161. Some(ipv4::Address::from_bytes(
  162. &self.0[IPV4_MAPPED_PREFIX_SIZE..],
  163. ))
  164. } else {
  165. None
  166. }
  167. }
  168. /// Helper function used to mask an address given a prefix.
  169. ///
  170. /// # Panics
  171. /// This function panics if `mask` is greater than 128.
  172. pub(super) fn mask(&self, mask: u8) -> [u8; ADDR_SIZE] {
  173. assert!(mask <= 128);
  174. let mut bytes = [0u8; ADDR_SIZE];
  175. let idx = (mask as usize) / 8;
  176. let modulus = (mask as usize) % 8;
  177. let (first, second) = self.0.split_at(idx);
  178. bytes[0..idx].copy_from_slice(first);
  179. if idx < ADDR_SIZE {
  180. let part = second[0];
  181. bytes[idx] = part & (!(0xff >> modulus) as u8);
  182. }
  183. bytes
  184. }
  185. /// The solicited node for the given unicast address.
  186. ///
  187. /// # Panics
  188. /// This function panics if the given address is not
  189. /// unicast.
  190. pub fn solicited_node(&self) -> Address {
  191. assert!(self.is_unicast());
  192. Address([
  193. 0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xFF,
  194. self.0[13], self.0[14], self.0[15],
  195. ])
  196. }
  197. /// Convert to an `IpAddress`.
  198. ///
  199. /// Same as `.into()`, but works in `const`.
  200. pub const fn into_address(self) -> super::IpAddress {
  201. super::IpAddress::Ipv6(self)
  202. }
  203. }
  204. #[cfg(feature = "std")]
  205. impl From<::std::net::Ipv6Addr> for Address {
  206. fn from(x: ::std::net::Ipv6Addr) -> Address {
  207. Address(x.octets())
  208. }
  209. }
  210. #[cfg(feature = "std")]
  211. impl From<Address> for ::std::net::Ipv6Addr {
  212. fn from(Address(x): Address) -> ::std::net::Ipv6Addr {
  213. x.into()
  214. }
  215. }
  216. impl fmt::Display for Address {
  217. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  218. if self.is_ipv4_mapped() {
  219. return write!(
  220. f,
  221. "::ffff:{}.{}.{}.{}",
  222. self.0[IPV4_MAPPED_PREFIX_SIZE + 0],
  223. self.0[IPV4_MAPPED_PREFIX_SIZE + 1],
  224. self.0[IPV4_MAPPED_PREFIX_SIZE + 2],
  225. self.0[IPV4_MAPPED_PREFIX_SIZE + 3]
  226. );
  227. }
  228. // The string representation of an IPv6 address should
  229. // collapse a series of 16 bit sections that evaluate
  230. // to 0 to "::"
  231. //
  232. // See https://tools.ietf.org/html/rfc4291#section-2.2
  233. // for details.
  234. enum State {
  235. Head,
  236. HeadBody,
  237. Tail,
  238. TailBody,
  239. }
  240. let mut words = [0u16; 8];
  241. self.write_parts(&mut words);
  242. let mut state = State::Head;
  243. for word in words.iter() {
  244. state = match (*word, &state) {
  245. // Once a u16 equal to zero write a double colon and
  246. // skip to the next non-zero u16.
  247. (0, &State::Head) | (0, &State::HeadBody) => {
  248. write!(f, "::")?;
  249. State::Tail
  250. }
  251. // Continue iterating without writing any characters until
  252. // we hit a non-zero value.
  253. (0, &State::Tail) => State::Tail,
  254. // When the state is Head or Tail write a u16 in hexadecimal
  255. // without the leading colon if the value is not 0.
  256. (_, &State::Head) => {
  257. write!(f, "{word:x}")?;
  258. State::HeadBody
  259. }
  260. (_, &State::Tail) => {
  261. write!(f, "{word:x}")?;
  262. State::TailBody
  263. }
  264. // Write the u16 with a leading colon when parsing a value
  265. // that isn't the first in a section
  266. (_, &State::HeadBody) | (_, &State::TailBody) => {
  267. write!(f, ":{word:x}")?;
  268. state
  269. }
  270. }
  271. }
  272. Ok(())
  273. }
  274. }
  275. #[cfg(feature = "defmt")]
  276. impl defmt::Format for Address {
  277. fn format(&self, f: defmt::Formatter) {
  278. if self.is_ipv4_mapped() {
  279. return defmt::write!(
  280. f,
  281. "::ffff:{}.{}.{}.{}",
  282. self.0[IPV4_MAPPED_PREFIX_SIZE + 0],
  283. self.0[IPV4_MAPPED_PREFIX_SIZE + 1],
  284. self.0[IPV4_MAPPED_PREFIX_SIZE + 2],
  285. self.0[IPV4_MAPPED_PREFIX_SIZE + 3]
  286. );
  287. }
  288. // The string representation of an IPv6 address should
  289. // collapse a series of 16 bit sections that evaluate
  290. // to 0 to "::"
  291. //
  292. // See https://tools.ietf.org/html/rfc4291#section-2.2
  293. // for details.
  294. enum State {
  295. Head,
  296. HeadBody,
  297. Tail,
  298. TailBody,
  299. }
  300. let mut words = [0u16; 8];
  301. self.write_parts(&mut words);
  302. let mut state = State::Head;
  303. for word in words.iter() {
  304. state = match (*word, &state) {
  305. // Once a u16 equal to zero write a double colon and
  306. // skip to the next non-zero u16.
  307. (0, &State::Head) | (0, &State::HeadBody) => {
  308. defmt::write!(f, "::");
  309. State::Tail
  310. }
  311. // Continue iterating without writing any characters until
  312. // we hit a non-zero value.
  313. (0, &State::Tail) => State::Tail,
  314. // When the state is Head or Tail write a u16 in hexadecimal
  315. // without the leading colon if the value is not 0.
  316. (_, &State::Head) => {
  317. defmt::write!(f, "{:x}", word);
  318. State::HeadBody
  319. }
  320. (_, &State::Tail) => {
  321. defmt::write!(f, "{:x}", word);
  322. State::TailBody
  323. }
  324. // Write the u16 with a leading colon when parsing a value
  325. // that isn't the first in a section
  326. (_, &State::HeadBody) | (_, &State::TailBody) => {
  327. defmt::write!(f, ":{:x}", word);
  328. state
  329. }
  330. }
  331. }
  332. }
  333. }
  334. #[cfg(feature = "proto-ipv4")]
  335. /// Convert the given IPv4 address into a IPv4-mapped IPv6 address
  336. impl From<ipv4::Address> for Address {
  337. fn from(address: ipv4::Address) -> Self {
  338. let mut b = [0_u8; ADDR_SIZE];
  339. b[..Self::IPV4_MAPPED_PREFIX.len()].copy_from_slice(&Self::IPV4_MAPPED_PREFIX);
  340. b[Self::IPV4_MAPPED_PREFIX.len()..].copy_from_slice(&address.0);
  341. Self(b)
  342. }
  343. }
  344. /// A specification of an IPv6 CIDR block, containing an address and a variable-length
  345. /// subnet masking prefix length.
  346. #[derive(Debug, Hash, PartialEq, Eq, PartialOrd, Ord, Clone, Copy, Default)]
  347. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  348. pub struct Cidr {
  349. address: Address,
  350. prefix_len: u8,
  351. }
  352. impl Cidr {
  353. /// The [solicited node prefix].
  354. ///
  355. /// [solicited node prefix]: https://tools.ietf.org/html/rfc4291#section-2.7.1
  356. pub const SOLICITED_NODE_PREFIX: Cidr = Cidr {
  357. address: Address([
  358. 0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xff, 0x00,
  359. 0x00, 0x00,
  360. ]),
  361. prefix_len: 104,
  362. };
  363. /// Create an IPv6 CIDR block from the given address and prefix length.
  364. ///
  365. /// # Panics
  366. /// This function panics if the prefix length is larger than 128.
  367. pub const fn new(address: Address, prefix_len: u8) -> Cidr {
  368. assert!(prefix_len <= 128);
  369. Cidr {
  370. address,
  371. prefix_len,
  372. }
  373. }
  374. /// Return the address of this IPv6 CIDR block.
  375. pub const fn address(&self) -> Address {
  376. self.address
  377. }
  378. /// Return the prefix length of this IPv6 CIDR block.
  379. pub const fn prefix_len(&self) -> u8 {
  380. self.prefix_len
  381. }
  382. /// Query whether the subnetwork described by this IPv6 CIDR block contains
  383. /// the given address.
  384. pub fn contains_addr(&self, addr: &Address) -> bool {
  385. // right shift by 128 is not legal
  386. if self.prefix_len == 0 {
  387. return true;
  388. }
  389. self.address.mask(self.prefix_len) == addr.mask(self.prefix_len)
  390. }
  391. /// Query whether the subnetwork described by this IPV6 CIDR block contains
  392. /// the subnetwork described by the given IPv6 CIDR block.
  393. pub fn contains_subnet(&self, subnet: &Cidr) -> bool {
  394. self.prefix_len <= subnet.prefix_len && self.contains_addr(&subnet.address)
  395. }
  396. }
  397. impl fmt::Display for Cidr {
  398. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  399. // https://tools.ietf.org/html/rfc4291#section-2.3
  400. write!(f, "{}/{}", self.address, self.prefix_len)
  401. }
  402. }
  403. /// A read/write wrapper around an Internet Protocol version 6 packet buffer.
  404. #[derive(Debug, PartialEq, Eq, Clone)]
  405. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  406. pub struct Packet<T: AsRef<[u8]>> {
  407. buffer: T,
  408. }
  409. // Ranges and constants describing the IPv6 header
  410. //
  411. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  412. // |Version| Traffic Class | Flow Label |
  413. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  414. // | Payload Length | Next Header | Hop Limit |
  415. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  416. // | |
  417. // + +
  418. // | |
  419. // + Source Address +
  420. // | |
  421. // + +
  422. // | |
  423. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  424. // | |
  425. // + +
  426. // | |
  427. // + Destination Address +
  428. // | |
  429. // + +
  430. // | |
  431. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  432. //
  433. // See https://tools.ietf.org/html/rfc2460#section-3 for details.
  434. mod field {
  435. use crate::wire::field::*;
  436. // 4-bit version number, 8-bit traffic class, and the
  437. // 20-bit flow label.
  438. pub const VER_TC_FLOW: Field = 0..4;
  439. // 16-bit value representing the length of the payload.
  440. // Note: Options are included in this length.
  441. pub const LENGTH: Field = 4..6;
  442. // 8-bit value identifying the type of header following this
  443. // one. Note: The same numbers are used in IPv4.
  444. pub const NXT_HDR: usize = 6;
  445. // 8-bit value decremented by each node that forwards this
  446. // packet. The packet is discarded when the value is 0.
  447. pub const HOP_LIMIT: usize = 7;
  448. // IPv6 address of the source node.
  449. pub const SRC_ADDR: Field = 8..24;
  450. // IPv6 address of the destination node.
  451. pub const DST_ADDR: Field = 24..40;
  452. }
  453. /// Length of an IPv6 header.
  454. pub const HEADER_LEN: usize = field::DST_ADDR.end;
  455. impl<T: AsRef<[u8]>> Packet<T> {
  456. /// Create a raw octet buffer with an IPv6 packet structure.
  457. #[inline]
  458. pub const fn new_unchecked(buffer: T) -> Packet<T> {
  459. Packet { buffer }
  460. }
  461. /// Shorthand for a combination of [new_unchecked] and [check_len].
  462. ///
  463. /// [new_unchecked]: #method.new_unchecked
  464. /// [check_len]: #method.check_len
  465. #[inline]
  466. pub fn new_checked(buffer: T) -> Result<Packet<T>> {
  467. let packet = Self::new_unchecked(buffer);
  468. packet.check_len()?;
  469. Ok(packet)
  470. }
  471. /// Ensure that no accessor method will panic if called.
  472. /// Returns `Err(Error)` if the buffer is too short.
  473. ///
  474. /// The result of this check is invalidated by calling [set_payload_len].
  475. ///
  476. /// [set_payload_len]: #method.set_payload_len
  477. #[inline]
  478. pub fn check_len(&self) -> Result<()> {
  479. let len = self.buffer.as_ref().len();
  480. if len < field::DST_ADDR.end || len < self.total_len() {
  481. Err(Error)
  482. } else {
  483. Ok(())
  484. }
  485. }
  486. /// Consume the packet, returning the underlying buffer.
  487. #[inline]
  488. pub fn into_inner(self) -> T {
  489. self.buffer
  490. }
  491. /// Return the header length.
  492. #[inline]
  493. pub const fn header_len(&self) -> usize {
  494. // This is not a strictly necessary function, but it makes
  495. // code more readable.
  496. field::DST_ADDR.end
  497. }
  498. /// Return the version field.
  499. #[inline]
  500. pub fn version(&self) -> u8 {
  501. let data = self.buffer.as_ref();
  502. data[field::VER_TC_FLOW.start] >> 4
  503. }
  504. /// Return the traffic class.
  505. #[inline]
  506. pub fn traffic_class(&self) -> u8 {
  507. let data = self.buffer.as_ref();
  508. ((NetworkEndian::read_u16(&data[0..2]) & 0x0ff0) >> 4) as u8
  509. }
  510. /// Return the flow label field.
  511. #[inline]
  512. pub fn flow_label(&self) -> u32 {
  513. let data = self.buffer.as_ref();
  514. NetworkEndian::read_u24(&data[1..4]) & 0x000fffff
  515. }
  516. /// Return the payload length field.
  517. #[inline]
  518. pub fn payload_len(&self) -> u16 {
  519. let data = self.buffer.as_ref();
  520. NetworkEndian::read_u16(&data[field::LENGTH])
  521. }
  522. /// Return the payload length added to the known header length.
  523. #[inline]
  524. pub fn total_len(&self) -> usize {
  525. self.header_len() + self.payload_len() as usize
  526. }
  527. /// Return the next header field.
  528. #[inline]
  529. pub fn next_header(&self) -> Protocol {
  530. let data = self.buffer.as_ref();
  531. Protocol::from(data[field::NXT_HDR])
  532. }
  533. /// Return the hop limit field.
  534. #[inline]
  535. pub fn hop_limit(&self) -> u8 {
  536. let data = self.buffer.as_ref();
  537. data[field::HOP_LIMIT]
  538. }
  539. /// Return the source address field.
  540. #[inline]
  541. pub fn src_addr(&self) -> Address {
  542. let data = self.buffer.as_ref();
  543. Address::from_bytes(&data[field::SRC_ADDR])
  544. }
  545. /// Return the destination address field.
  546. #[inline]
  547. pub fn dst_addr(&self) -> Address {
  548. let data = self.buffer.as_ref();
  549. Address::from_bytes(&data[field::DST_ADDR])
  550. }
  551. }
  552. impl<'a, T: AsRef<[u8]> + ?Sized> Packet<&'a T> {
  553. /// Return a pointer to the payload.
  554. #[inline]
  555. pub fn payload(&self) -> &'a [u8] {
  556. let data = self.buffer.as_ref();
  557. let range = self.header_len()..self.total_len();
  558. &data[range]
  559. }
  560. }
  561. impl<T: AsRef<[u8]> + AsMut<[u8]>> Packet<T> {
  562. /// Set the version field.
  563. #[inline]
  564. pub fn set_version(&mut self, value: u8) {
  565. let data = self.buffer.as_mut();
  566. // Make sure to retain the lower order bits which contain
  567. // the higher order bits of the traffic class
  568. data[0] = (data[0] & 0x0f) | ((value & 0x0f) << 4);
  569. }
  570. /// Set the traffic class field.
  571. #[inline]
  572. pub fn set_traffic_class(&mut self, value: u8) {
  573. let data = self.buffer.as_mut();
  574. // Put the higher order 4-bits of value in the lower order
  575. // 4-bits of the first byte
  576. data[0] = (data[0] & 0xf0) | ((value & 0xf0) >> 4);
  577. // Put the lower order 4-bits of value in the higher order
  578. // 4-bits of the second byte
  579. data[1] = (data[1] & 0x0f) | ((value & 0x0f) << 4);
  580. }
  581. /// Set the flow label field.
  582. #[inline]
  583. pub fn set_flow_label(&mut self, value: u32) {
  584. let data = self.buffer.as_mut();
  585. // Retain the lower order 4-bits of the traffic class
  586. let raw = (((data[1] & 0xf0) as u32) << 16) | (value & 0x0fffff);
  587. NetworkEndian::write_u24(&mut data[1..4], raw);
  588. }
  589. /// Set the payload length field.
  590. #[inline]
  591. pub fn set_payload_len(&mut self, value: u16) {
  592. let data = self.buffer.as_mut();
  593. NetworkEndian::write_u16(&mut data[field::LENGTH], value);
  594. }
  595. /// Set the next header field.
  596. #[inline]
  597. pub fn set_next_header(&mut self, value: Protocol) {
  598. let data = self.buffer.as_mut();
  599. data[field::NXT_HDR] = value.into();
  600. }
  601. /// Set the hop limit field.
  602. #[inline]
  603. pub fn set_hop_limit(&mut self, value: u8) {
  604. let data = self.buffer.as_mut();
  605. data[field::HOP_LIMIT] = value;
  606. }
  607. /// Set the source address field.
  608. #[inline]
  609. pub fn set_src_addr(&mut self, value: Address) {
  610. let data = self.buffer.as_mut();
  611. data[field::SRC_ADDR].copy_from_slice(value.as_bytes());
  612. }
  613. /// Set the destination address field.
  614. #[inline]
  615. pub fn set_dst_addr(&mut self, value: Address) {
  616. let data = self.buffer.as_mut();
  617. data[field::DST_ADDR].copy_from_slice(value.as_bytes());
  618. }
  619. /// Return a mutable pointer to the payload.
  620. #[inline]
  621. pub fn payload_mut(&mut self) -> &mut [u8] {
  622. let range = self.header_len()..self.total_len();
  623. let data = self.buffer.as_mut();
  624. &mut data[range]
  625. }
  626. }
  627. impl<'a, T: AsRef<[u8]> + ?Sized> fmt::Display for Packet<&'a T> {
  628. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  629. match Repr::parse(self) {
  630. Ok(repr) => write!(f, "{repr}"),
  631. Err(err) => {
  632. write!(f, "IPv6 ({err})")?;
  633. Ok(())
  634. }
  635. }
  636. }
  637. }
  638. impl<T: AsRef<[u8]>> AsRef<[u8]> for Packet<T> {
  639. fn as_ref(&self) -> &[u8] {
  640. self.buffer.as_ref()
  641. }
  642. }
  643. /// A high-level representation of an Internet Protocol version 6 packet header.
  644. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  645. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  646. pub struct Repr {
  647. /// IPv6 address of the source node.
  648. pub src_addr: Address,
  649. /// IPv6 address of the destination node.
  650. pub dst_addr: Address,
  651. /// Protocol contained in the next header.
  652. pub next_header: Protocol,
  653. /// Length of the payload including the extension headers.
  654. pub payload_len: usize,
  655. /// The 8-bit hop limit field.
  656. pub hop_limit: u8,
  657. }
  658. impl Repr {
  659. /// Parse an Internet Protocol version 6 packet and return a high-level representation.
  660. pub fn parse<T: AsRef<[u8]> + ?Sized>(packet: &Packet<&T>) -> Result<Repr> {
  661. // Ensure basic accessors will work
  662. packet.check_len()?;
  663. if packet.version() != 6 {
  664. return Err(Error);
  665. }
  666. Ok(Repr {
  667. src_addr: packet.src_addr(),
  668. dst_addr: packet.dst_addr(),
  669. next_header: packet.next_header(),
  670. payload_len: packet.payload_len() as usize,
  671. hop_limit: packet.hop_limit(),
  672. })
  673. }
  674. /// Return the length of a header that will be emitted from this high-level representation.
  675. pub const fn buffer_len(&self) -> usize {
  676. // This function is not strictly necessary, but it can make client code more readable.
  677. field::DST_ADDR.end
  678. }
  679. /// Emit a high-level representation into an Internet Protocol version 6 packet.
  680. pub fn emit<T: AsRef<[u8]> + AsMut<[u8]>>(&self, packet: &mut Packet<T>) {
  681. // Make no assumptions about the original state of the packet buffer.
  682. // Make sure to set every byte.
  683. packet.set_version(6);
  684. packet.set_traffic_class(0);
  685. packet.set_flow_label(0);
  686. packet.set_payload_len(self.payload_len as u16);
  687. packet.set_hop_limit(self.hop_limit);
  688. packet.set_next_header(self.next_header);
  689. packet.set_src_addr(self.src_addr);
  690. packet.set_dst_addr(self.dst_addr);
  691. }
  692. }
  693. impl fmt::Display for Repr {
  694. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  695. write!(
  696. f,
  697. "IPv6 src={} dst={} nxt_hdr={} hop_limit={}",
  698. self.src_addr, self.dst_addr, self.next_header, self.hop_limit
  699. )
  700. }
  701. }
  702. use crate::wire::pretty_print::{PrettyIndent, PrettyPrint};
  703. // TODO: This is very similar to the implementation for IPv4. Make
  704. // a way to have less copy and pasted code here.
  705. impl<T: AsRef<[u8]>> PrettyPrint for Packet<T> {
  706. fn pretty_print(
  707. buffer: &dyn AsRef<[u8]>,
  708. f: &mut fmt::Formatter,
  709. indent: &mut PrettyIndent,
  710. ) -> fmt::Result {
  711. let (ip_repr, payload) = match Packet::new_checked(buffer) {
  712. Err(err) => return write!(f, "{indent}({err})"),
  713. Ok(ip_packet) => match Repr::parse(&ip_packet) {
  714. Err(_) => return Ok(()),
  715. Ok(ip_repr) => {
  716. write!(f, "{indent}{ip_repr}")?;
  717. (ip_repr, ip_packet.payload())
  718. }
  719. },
  720. };
  721. pretty_print_ip_payload(f, indent, ip_repr, payload)
  722. }
  723. }
  724. #[cfg(test)]
  725. mod test {
  726. use super::Error;
  727. use super::{Address, Cidr};
  728. use super::{Packet, Protocol, Repr};
  729. use crate::wire::pretty_print::PrettyPrinter;
  730. #[cfg(feature = "proto-ipv4")]
  731. use crate::wire::ipv4::Address as Ipv4Address;
  732. static LINK_LOCAL_ADDR: Address = Address([
  733. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  734. 0x01,
  735. ]);
  736. #[test]
  737. fn test_basic_multicast() {
  738. assert!(!Address::LINK_LOCAL_ALL_ROUTERS.is_unspecified());
  739. assert!(Address::LINK_LOCAL_ALL_ROUTERS.is_multicast());
  740. assert!(!Address::LINK_LOCAL_ALL_ROUTERS.is_link_local());
  741. assert!(!Address::LINK_LOCAL_ALL_ROUTERS.is_loopback());
  742. assert!(!Address::LINK_LOCAL_ALL_NODES.is_unspecified());
  743. assert!(Address::LINK_LOCAL_ALL_NODES.is_multicast());
  744. assert!(!Address::LINK_LOCAL_ALL_NODES.is_link_local());
  745. assert!(!Address::LINK_LOCAL_ALL_NODES.is_loopback());
  746. }
  747. #[test]
  748. fn test_basic_link_local() {
  749. assert!(!LINK_LOCAL_ADDR.is_unspecified());
  750. assert!(!LINK_LOCAL_ADDR.is_multicast());
  751. assert!(LINK_LOCAL_ADDR.is_link_local());
  752. assert!(!LINK_LOCAL_ADDR.is_loopback());
  753. }
  754. #[test]
  755. fn test_basic_loopback() {
  756. assert!(!Address::LOOPBACK.is_unspecified());
  757. assert!(!Address::LOOPBACK.is_multicast());
  758. assert!(!Address::LOOPBACK.is_link_local());
  759. assert!(Address::LOOPBACK.is_loopback());
  760. }
  761. #[test]
  762. fn test_address_format() {
  763. assert_eq!("ff02::1", format!("{}", Address::LINK_LOCAL_ALL_NODES));
  764. assert_eq!("fe80::1", format!("{LINK_LOCAL_ADDR}"));
  765. assert_eq!(
  766. "fe80::7f00:0:1",
  767. format!(
  768. "{}",
  769. Address::new(0xfe80, 0, 0, 0, 0, 0x7f00, 0x0000, 0x0001)
  770. )
  771. );
  772. assert_eq!("::", format!("{}", Address::UNSPECIFIED));
  773. assert_eq!("::1", format!("{}", Address::LOOPBACK));
  774. #[cfg(feature = "proto-ipv4")]
  775. assert_eq!(
  776. "::ffff:192.168.1.1",
  777. format!("{}", Address::from(Ipv4Address::new(192, 168, 1, 1)))
  778. );
  779. }
  780. #[test]
  781. fn test_new() {
  782. assert_eq!(
  783. Address::new(0xff02, 0, 0, 0, 0, 0, 0, 1),
  784. Address::LINK_LOCAL_ALL_NODES
  785. );
  786. assert_eq!(
  787. Address::new(0xff02, 0, 0, 0, 0, 0, 0, 2),
  788. Address::LINK_LOCAL_ALL_ROUTERS
  789. );
  790. assert_eq!(Address::new(0, 0, 0, 0, 0, 0, 0, 1), Address::LOOPBACK);
  791. assert_eq!(Address::new(0, 0, 0, 0, 0, 0, 0, 0), Address::UNSPECIFIED);
  792. assert_eq!(Address::new(0xfe80, 0, 0, 0, 0, 0, 0, 1), LINK_LOCAL_ADDR);
  793. }
  794. #[test]
  795. fn test_from_parts() {
  796. assert_eq!(
  797. Address::from_parts(&[0xff02, 0, 0, 0, 0, 0, 0, 1]),
  798. Address::LINK_LOCAL_ALL_NODES
  799. );
  800. assert_eq!(
  801. Address::from_parts(&[0xff02, 0, 0, 0, 0, 0, 0, 2]),
  802. Address::LINK_LOCAL_ALL_ROUTERS
  803. );
  804. assert_eq!(
  805. Address::from_parts(&[0, 0, 0, 0, 0, 0, 0, 1]),
  806. Address::LOOPBACK
  807. );
  808. assert_eq!(
  809. Address::from_parts(&[0, 0, 0, 0, 0, 0, 0, 0]),
  810. Address::UNSPECIFIED
  811. );
  812. assert_eq!(
  813. Address::from_parts(&[0xfe80, 0, 0, 0, 0, 0, 0, 1]),
  814. LINK_LOCAL_ADDR
  815. );
  816. }
  817. #[test]
  818. fn test_write_parts() {
  819. let mut bytes = [0u16; 8];
  820. {
  821. Address::LOOPBACK.write_parts(&mut bytes);
  822. assert_eq!(Address::LOOPBACK, Address::from_parts(&bytes));
  823. }
  824. {
  825. Address::LINK_LOCAL_ALL_ROUTERS.write_parts(&mut bytes);
  826. assert_eq!(Address::LINK_LOCAL_ALL_ROUTERS, Address::from_parts(&bytes));
  827. }
  828. {
  829. LINK_LOCAL_ADDR.write_parts(&mut bytes);
  830. assert_eq!(LINK_LOCAL_ADDR, Address::from_parts(&bytes));
  831. }
  832. }
  833. #[test]
  834. fn test_mask() {
  835. let addr = Address::new(0x0123, 0x4567, 0x89ab, 0, 0, 0, 0, 1);
  836. assert_eq!(
  837. addr.mask(11),
  838. [0x01, 0x20, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
  839. );
  840. assert_eq!(
  841. addr.mask(15),
  842. [0x01, 0x22, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
  843. );
  844. assert_eq!(
  845. addr.mask(26),
  846. [0x01, 0x23, 0x45, 0x40, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
  847. );
  848. assert_eq!(
  849. addr.mask(128),
  850. [0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1]
  851. );
  852. assert_eq!(
  853. addr.mask(127),
  854. [0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
  855. );
  856. }
  857. #[cfg(feature = "proto-ipv4")]
  858. #[test]
  859. fn test_is_ipv4_mapped() {
  860. assert!(!Address::UNSPECIFIED.is_ipv4_mapped());
  861. assert!(Address::from(Ipv4Address::new(192, 168, 1, 1)).is_ipv4_mapped());
  862. }
  863. #[cfg(feature = "proto-ipv4")]
  864. #[test]
  865. fn test_as_ipv4() {
  866. assert_eq!(None, Address::UNSPECIFIED.as_ipv4());
  867. let ipv4 = Ipv4Address::new(192, 168, 1, 1);
  868. assert_eq!(Some(ipv4), Address::from(ipv4).as_ipv4());
  869. }
  870. #[cfg(feature = "proto-ipv4")]
  871. #[test]
  872. fn test_from_ipv4_address() {
  873. assert_eq!(
  874. Address([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff, 192, 168, 1, 1]),
  875. Address::from(Ipv4Address::new(192, 168, 1, 1))
  876. );
  877. assert_eq!(
  878. Address([0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff, 222, 1, 41, 90]),
  879. Address::from(Ipv4Address::new(222, 1, 41, 90))
  880. );
  881. }
  882. #[test]
  883. fn test_cidr() {
  884. // fe80::1/56
  885. // 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00,
  886. // 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
  887. let cidr = Cidr::new(LINK_LOCAL_ADDR, 56);
  888. let inside_subnet = [
  889. // fe80::2
  890. [
  891. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  892. 0x00, 0x02,
  893. ],
  894. // fe80::1122:3344:5566:7788
  895. [
  896. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66,
  897. 0x77, 0x88,
  898. ],
  899. // fe80::ff00:0:0:0
  900. [
  901. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0x00, 0x00, 0x00, 0x00, 0x00,
  902. 0x00, 0x00,
  903. ],
  904. // fe80::ff
  905. [
  906. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  907. 0x00, 0xff,
  908. ],
  909. ];
  910. let outside_subnet = [
  911. // fe80:0:0:101::1
  912. [
  913. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  914. 0x00, 0x01,
  915. ],
  916. // ::1
  917. [
  918. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  919. 0x00, 0x01,
  920. ],
  921. // ff02::1
  922. [
  923. 0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  924. 0x00, 0x01,
  925. ],
  926. // ff02::2
  927. [
  928. 0xff, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  929. 0x00, 0x02,
  930. ],
  931. ];
  932. let subnets = [
  933. // fe80::ffff:ffff:ffff:ffff/65
  934. (
  935. [
  936. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0xff, 0xff, 0xff,
  937. 0xff, 0xff, 0xff,
  938. ],
  939. 65,
  940. ),
  941. // fe80::1/128
  942. (
  943. [
  944. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  945. 0x00, 0x00, 0x01,
  946. ],
  947. 128,
  948. ),
  949. // fe80::1234:5678/96
  950. (
  951. [
  952. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x12,
  953. 0x34, 0x56, 0x78,
  954. ],
  955. 96,
  956. ),
  957. ];
  958. let not_subnets = [
  959. // fe80::101:ffff:ffff:ffff:ffff/55
  960. (
  961. [
  962. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0xff, 0xff, 0xff, 0xff, 0xff,
  963. 0xff, 0xff, 0xff,
  964. ],
  965. 55,
  966. ),
  967. // fe80::101:ffff:ffff:ffff:ffff/56
  968. (
  969. [
  970. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0xff, 0xff, 0xff, 0xff, 0xff,
  971. 0xff, 0xff, 0xff,
  972. ],
  973. 56,
  974. ),
  975. // fe80::101:ffff:ffff:ffff:ffff/57
  976. (
  977. [
  978. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x01, 0x01, 0xff, 0xff, 0xff, 0xff, 0xff,
  979. 0xff, 0xff, 0xff,
  980. ],
  981. 57,
  982. ),
  983. // ::1/128
  984. (
  985. [
  986. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  987. 0x00, 0x00, 0x01,
  988. ],
  989. 128,
  990. ),
  991. ];
  992. for addr in inside_subnet.iter().map(|a| Address::from_bytes(a)) {
  993. assert!(cidr.contains_addr(&addr));
  994. }
  995. for addr in outside_subnet.iter().map(|a| Address::from_bytes(a)) {
  996. assert!(!cidr.contains_addr(&addr));
  997. }
  998. for subnet in subnets.iter().map(|&(a, p)| Cidr::new(Address(a), p)) {
  999. assert!(cidr.contains_subnet(&subnet));
  1000. }
  1001. for subnet in not_subnets.iter().map(|&(a, p)| Cidr::new(Address(a), p)) {
  1002. assert!(!cidr.contains_subnet(&subnet));
  1003. }
  1004. let cidr_without_prefix = Cidr::new(LINK_LOCAL_ADDR, 0);
  1005. assert!(cidr_without_prefix.contains_addr(&Address::LOOPBACK));
  1006. }
  1007. #[test]
  1008. #[should_panic(expected = "length")]
  1009. fn test_from_bytes_too_long() {
  1010. let _ = Address::from_bytes(&[0u8; 15]);
  1011. }
  1012. #[test]
  1013. #[should_panic(expected = "data.len() >= 8")]
  1014. fn test_from_parts_too_long() {
  1015. let _ = Address::from_parts(&[0u16; 7]);
  1016. }
  1017. static REPR_PACKET_BYTES: [u8; 52] = [
  1018. 0x60, 0x00, 0x00, 0x00, 0x00, 0x0c, 0x11, 0x40, 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00,
  1019. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0xff, 0x02, 0x00, 0x00, 0x00, 0x00,
  1020. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x02, 0x00,
  1021. 0x0c, 0x02, 0x4e, 0xff, 0xff, 0xff, 0xff,
  1022. ];
  1023. static REPR_PAYLOAD_BYTES: [u8; 12] = [
  1024. 0x00, 0x01, 0x00, 0x02, 0x00, 0x0c, 0x02, 0x4e, 0xff, 0xff, 0xff, 0xff,
  1025. ];
  1026. const fn packet_repr() -> Repr {
  1027. Repr {
  1028. src_addr: Address([
  1029. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  1030. 0x00, 0x01,
  1031. ]),
  1032. dst_addr: Address::LINK_LOCAL_ALL_NODES,
  1033. next_header: Protocol::Udp,
  1034. payload_len: 12,
  1035. hop_limit: 64,
  1036. }
  1037. }
  1038. #[test]
  1039. fn test_packet_deconstruction() {
  1040. let packet = Packet::new_unchecked(&REPR_PACKET_BYTES[..]);
  1041. assert_eq!(packet.check_len(), Ok(()));
  1042. assert_eq!(packet.version(), 6);
  1043. assert_eq!(packet.traffic_class(), 0);
  1044. assert_eq!(packet.flow_label(), 0);
  1045. assert_eq!(packet.total_len(), 0x34);
  1046. assert_eq!(packet.payload_len() as usize, REPR_PAYLOAD_BYTES.len());
  1047. assert_eq!(packet.next_header(), Protocol::Udp);
  1048. assert_eq!(packet.hop_limit(), 0x40);
  1049. assert_eq!(
  1050. packet.src_addr(),
  1051. Address([
  1052. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  1053. 0x00, 0x01
  1054. ])
  1055. );
  1056. assert_eq!(packet.dst_addr(), Address::LINK_LOCAL_ALL_NODES);
  1057. assert_eq!(packet.payload(), &REPR_PAYLOAD_BYTES[..]);
  1058. }
  1059. #[test]
  1060. fn test_packet_construction() {
  1061. let mut bytes = [0xff; 52];
  1062. let mut packet = Packet::new_unchecked(&mut bytes[..]);
  1063. // Version, Traffic Class, and Flow Label are not
  1064. // byte aligned. make sure the setters and getters
  1065. // do not interfere with each other.
  1066. packet.set_version(6);
  1067. assert_eq!(packet.version(), 6);
  1068. packet.set_traffic_class(0x99);
  1069. assert_eq!(packet.version(), 6);
  1070. assert_eq!(packet.traffic_class(), 0x99);
  1071. packet.set_flow_label(0x54321);
  1072. assert_eq!(packet.traffic_class(), 0x99);
  1073. assert_eq!(packet.flow_label(), 0x54321);
  1074. packet.set_payload_len(0xc);
  1075. packet.set_next_header(Protocol::Udp);
  1076. packet.set_hop_limit(0xfe);
  1077. packet.set_src_addr(Address::LINK_LOCAL_ALL_ROUTERS);
  1078. packet.set_dst_addr(Address::LINK_LOCAL_ALL_NODES);
  1079. packet
  1080. .payload_mut()
  1081. .copy_from_slice(&REPR_PAYLOAD_BYTES[..]);
  1082. let mut expected_bytes = [
  1083. 0x69, 0x95, 0x43, 0x21, 0x00, 0x0c, 0x11, 0xfe, 0xff, 0x02, 0x00, 0x00, 0x00, 0x00,
  1084. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x02, 0x00, 0x00,
  1085. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00,
  1086. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  1087. ];
  1088. let start = expected_bytes.len() - REPR_PAYLOAD_BYTES.len();
  1089. expected_bytes[start..].copy_from_slice(&REPR_PAYLOAD_BYTES[..]);
  1090. assert_eq!(packet.check_len(), Ok(()));
  1091. assert_eq!(&*packet.into_inner(), &expected_bytes[..]);
  1092. }
  1093. #[test]
  1094. fn test_overlong() {
  1095. let mut bytes = vec![];
  1096. bytes.extend(&REPR_PACKET_BYTES[..]);
  1097. bytes.push(0);
  1098. assert_eq!(
  1099. Packet::new_unchecked(&bytes).payload().len(),
  1100. REPR_PAYLOAD_BYTES.len()
  1101. );
  1102. assert_eq!(
  1103. Packet::new_unchecked(&mut bytes).payload_mut().len(),
  1104. REPR_PAYLOAD_BYTES.len()
  1105. );
  1106. }
  1107. #[test]
  1108. fn test_total_len_overflow() {
  1109. let mut bytes = vec![];
  1110. bytes.extend(&REPR_PACKET_BYTES[..]);
  1111. Packet::new_unchecked(&mut bytes).set_payload_len(0x80);
  1112. assert_eq!(Packet::new_checked(&bytes).unwrap_err(), Error);
  1113. }
  1114. #[test]
  1115. fn test_repr_parse_valid() {
  1116. let packet = Packet::new_unchecked(&REPR_PACKET_BYTES[..]);
  1117. let repr = Repr::parse(&packet).unwrap();
  1118. assert_eq!(repr, packet_repr());
  1119. }
  1120. #[test]
  1121. fn test_repr_parse_bad_version() {
  1122. let mut bytes = vec![0; 40];
  1123. let mut packet = Packet::new_unchecked(&mut bytes[..]);
  1124. packet.set_version(4);
  1125. packet.set_payload_len(0);
  1126. let packet = Packet::new_unchecked(&*packet.into_inner());
  1127. assert_eq!(Repr::parse(&packet), Err(Error));
  1128. }
  1129. #[test]
  1130. fn test_repr_parse_smaller_than_header() {
  1131. let mut bytes = vec![0; 40];
  1132. let mut packet = Packet::new_unchecked(&mut bytes[..]);
  1133. packet.set_version(6);
  1134. packet.set_payload_len(39);
  1135. let packet = Packet::new_unchecked(&*packet.into_inner());
  1136. assert_eq!(Repr::parse(&packet), Err(Error));
  1137. }
  1138. #[test]
  1139. fn test_repr_parse_smaller_than_payload() {
  1140. let mut bytes = vec![0; 40];
  1141. let mut packet = Packet::new_unchecked(&mut bytes[..]);
  1142. packet.set_version(6);
  1143. packet.set_payload_len(1);
  1144. let packet = Packet::new_unchecked(&*packet.into_inner());
  1145. assert_eq!(Repr::parse(&packet), Err(Error));
  1146. }
  1147. #[test]
  1148. fn test_basic_repr_emit() {
  1149. let repr = packet_repr();
  1150. let mut bytes = vec![0xff; repr.buffer_len() + REPR_PAYLOAD_BYTES.len()];
  1151. let mut packet = Packet::new_unchecked(&mut bytes);
  1152. repr.emit(&mut packet);
  1153. packet.payload_mut().copy_from_slice(&REPR_PAYLOAD_BYTES);
  1154. assert_eq!(&*packet.into_inner(), &REPR_PACKET_BYTES[..]);
  1155. }
  1156. #[test]
  1157. fn test_pretty_print() {
  1158. assert_eq!(
  1159. format!(
  1160. "{}",
  1161. PrettyPrinter::<Packet<&'static [u8]>>::new("\n", &&REPR_PACKET_BYTES[..])
  1162. ),
  1163. "\nIPv6 src=fe80::1 dst=ff02::1 nxt_hdr=UDP hop_limit=64\n \\ UDP src=1 dst=2 len=4"
  1164. );
  1165. }
  1166. }