ipv6fragment.rs 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312
  1. use super::{Error, Result};
  2. use core::fmt;
  3. use byteorder::{ByteOrder, NetworkEndian};
  4. pub use super::IpProtocol as Protocol;
  5. /// A read/write wrapper around an IPv6 Fragment Header.
  6. #[derive(Debug, PartialEq, Eq)]
  7. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  8. pub struct Header<T: AsRef<[u8]>> {
  9. buffer: T,
  10. }
  11. // Format of the Fragment Header
  12. //
  13. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  14. // | Next Header | Reserved | Fragment Offset |Res|M|
  15. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  16. // | Identification |
  17. // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  18. //
  19. // See https://tools.ietf.org/html/rfc8200#section-4.5 for details.
  20. mod field {
  21. use crate::wire::field::*;
  22. // 8-bit identifier of the header immediately following this header.
  23. pub const NXT_HDR: usize = 0;
  24. // 8-bit reserved field.
  25. pub const RESERVED: usize = 1;
  26. // 16-bit field containing the fragment offset, reserved and more fragments values.
  27. pub const FR_OF_M: Field = 2..4;
  28. // 32-bit field identifying the fragmented packet
  29. pub const IDENT: Field = 4..8;
  30. }
  31. impl<T: AsRef<[u8]>> Header<T> {
  32. /// Create a raw octet buffer with an IPv6 Fragment Header structure.
  33. pub fn new_unchecked(buffer: T) -> Header<T> {
  34. Header { buffer }
  35. }
  36. /// Shorthand for a combination of [new_unchecked] and [check_len].
  37. ///
  38. /// [new_unchecked]: #method.new_unchecked
  39. /// [check_len]: #method.check_len
  40. pub fn new_checked(buffer: T) -> Result<Header<T>> {
  41. let header = Self::new_unchecked(buffer);
  42. header.check_len()?;
  43. Ok(header)
  44. }
  45. /// Ensure that no accessor method will panic if called.
  46. /// Returns `Err(Error)` if the buffer is too short.
  47. pub fn check_len(&self) -> Result<()> {
  48. let data = self.buffer.as_ref();
  49. let len = data.len();
  50. if len < field::IDENT.end {
  51. Err(Error)
  52. } else {
  53. Ok(())
  54. }
  55. }
  56. /// Consume the header, returning the underlying buffer.
  57. pub fn into_inner(self) -> T {
  58. self.buffer
  59. }
  60. /// Return the next header field.
  61. #[inline]
  62. pub fn next_header(&self) -> Protocol {
  63. let data = self.buffer.as_ref();
  64. Protocol::from(data[field::NXT_HDR])
  65. }
  66. /// Return the fragment offset field.
  67. #[inline]
  68. pub fn frag_offset(&self) -> u16 {
  69. let data = self.buffer.as_ref();
  70. NetworkEndian::read_u16(&data[field::FR_OF_M]) >> 3
  71. }
  72. /// Return more fragment flag field.
  73. #[inline]
  74. pub fn more_frags(&self) -> bool {
  75. let data = self.buffer.as_ref();
  76. (data[3] & 0x1) == 1
  77. }
  78. /// Return the fragment identification value field.
  79. #[inline]
  80. pub fn ident(&self) -> u32 {
  81. let data = self.buffer.as_ref();
  82. NetworkEndian::read_u32(&data[field::IDENT])
  83. }
  84. }
  85. impl<T: AsRef<[u8]> + AsMut<[u8]>> Header<T> {
  86. /// Set the next header field.
  87. #[inline]
  88. pub fn set_next_header(&mut self, value: Protocol) {
  89. let data = self.buffer.as_mut();
  90. data[field::NXT_HDR] = value.into();
  91. }
  92. /// Set reserved fields.
  93. ///
  94. /// Set 8-bit reserved field after the next header field.
  95. /// Set 2-bit reserved field between fragment offset and more fragments.
  96. #[inline]
  97. pub fn clear_reserved(&mut self) {
  98. let data = self.buffer.as_mut();
  99. data[field::RESERVED] = 0;
  100. // Retain the higher order 5 bits and lower order 1 bit
  101. data[3] &= 0xf9;
  102. }
  103. /// Set the fragment offset field.
  104. #[inline]
  105. pub fn set_frag_offset(&mut self, value: u16) {
  106. let data = self.buffer.as_mut();
  107. // Retain the lower order 3 bits
  108. let raw = ((value & 0x1fff) << 3) | ((data[3] & 0x7) as u16);
  109. NetworkEndian::write_u16(&mut data[field::FR_OF_M], raw);
  110. }
  111. /// Set the more fragments flag field.
  112. #[inline]
  113. pub fn set_more_frags(&mut self, value: bool) {
  114. let data = self.buffer.as_mut();
  115. // Retain the high order 7 bits
  116. let raw = (data[3] & 0xfe) | (value as u8 & 0x1);
  117. data[3] = raw;
  118. }
  119. /// Set the fragmentation identification field.
  120. #[inline]
  121. pub fn set_ident(&mut self, value: u32) {
  122. let data = self.buffer.as_mut();
  123. NetworkEndian::write_u32(&mut data[field::IDENT], value);
  124. }
  125. }
  126. impl<'a, T: AsRef<[u8]> + ?Sized> fmt::Display for Header<&'a T> {
  127. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  128. match Repr::parse(self) {
  129. Ok(repr) => write!(f, "{}", repr),
  130. Err(err) => {
  131. write!(f, "IPv6 Fragment ({})", err)?;
  132. Ok(())
  133. }
  134. }
  135. }
  136. }
  137. /// A high-level representation of an IPv6 Fragment header.
  138. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  139. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  140. pub struct Repr {
  141. /// The type of header immediately following the Fragment header.
  142. pub next_header: Protocol,
  143. /// The offset of the data following this header, relative to the start of the Fragmentable
  144. /// Part of the original packet.
  145. pub frag_offset: u16,
  146. /// When there are more fragments following this header
  147. pub more_frags: bool,
  148. /// The identification for every packet that is fragmented.
  149. pub ident: u32,
  150. }
  151. impl Repr {
  152. /// Parse an IPv6 Fragment Header and return a high-level representation.
  153. pub fn parse<T>(header: &Header<&T>) -> Result<Repr>
  154. where
  155. T: AsRef<[u8]> + ?Sized,
  156. {
  157. Ok(Repr {
  158. next_header: header.next_header(),
  159. frag_offset: header.frag_offset(),
  160. more_frags: header.more_frags(),
  161. ident: header.ident(),
  162. })
  163. }
  164. /// Return the length, in bytes, of a header that will be emitted from this high-level
  165. /// representation.
  166. pub fn buffer_len(&self) -> usize {
  167. field::IDENT.end
  168. }
  169. /// Emit a high-level representation into an IPv6 Fragment Header.
  170. pub fn emit<T: AsRef<[u8]> + AsMut<[u8]> + ?Sized>(&self, header: &mut Header<&mut T>) {
  171. header.set_next_header(self.next_header);
  172. header.clear_reserved();
  173. header.set_frag_offset(self.frag_offset);
  174. header.set_more_frags(self.more_frags);
  175. header.set_ident(self.ident);
  176. }
  177. }
  178. impl fmt::Display for Repr {
  179. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  180. write!(
  181. f,
  182. "IPv6 Fragment next_hdr={} offset={} more={} ident={}",
  183. self.next_header, self.frag_offset, self.more_frags, self.ident
  184. )
  185. }
  186. }
  187. #[cfg(test)]
  188. mod test {
  189. use super::*;
  190. // A Fragment Header with more fragments remaining
  191. static BYTES_HEADER_MORE_FRAG: [u8; 8] = [0x6, 0x0, 0x0, 0x1, 0x0, 0x0, 0x30, 0x39];
  192. // A Fragment Header with no more fragments remaining
  193. static BYTES_HEADER_LAST_FRAG: [u8; 8] = [0x6, 0x0, 0xa, 0x0, 0x0, 0x1, 0x9, 0x32];
  194. #[test]
  195. fn test_check_len() {
  196. // less than 8 bytes
  197. assert_eq!(
  198. Err(Error),
  199. Header::new_unchecked(&BYTES_HEADER_MORE_FRAG[..7]).check_len()
  200. );
  201. // valid
  202. assert_eq!(
  203. Ok(()),
  204. Header::new_unchecked(&BYTES_HEADER_MORE_FRAG).check_len()
  205. );
  206. }
  207. #[test]
  208. fn test_header_deconstruct() {
  209. let header = Header::new_unchecked(&BYTES_HEADER_MORE_FRAG);
  210. assert_eq!(header.next_header(), Protocol::Tcp);
  211. assert_eq!(header.frag_offset(), 0);
  212. assert!(header.more_frags());
  213. assert_eq!(header.ident(), 12345);
  214. let header = Header::new_unchecked(&BYTES_HEADER_LAST_FRAG);
  215. assert_eq!(header.next_header(), Protocol::Tcp);
  216. assert_eq!(header.frag_offset(), 320);
  217. assert!(!header.more_frags());
  218. assert_eq!(header.ident(), 67890);
  219. }
  220. #[test]
  221. fn test_repr_parse_valid() {
  222. let header = Header::new_unchecked(&BYTES_HEADER_MORE_FRAG);
  223. let repr = Repr::parse(&header).unwrap();
  224. assert_eq!(
  225. repr,
  226. Repr {
  227. next_header: Protocol::Tcp,
  228. frag_offset: 0,
  229. more_frags: true,
  230. ident: 12345
  231. }
  232. );
  233. let header = Header::new_unchecked(&BYTES_HEADER_LAST_FRAG);
  234. let repr = Repr::parse(&header).unwrap();
  235. assert_eq!(
  236. repr,
  237. Repr {
  238. next_header: Protocol::Tcp,
  239. frag_offset: 320,
  240. more_frags: false,
  241. ident: 67890
  242. }
  243. );
  244. }
  245. #[test]
  246. fn test_repr_emit() {
  247. let repr = Repr {
  248. next_header: Protocol::Tcp,
  249. frag_offset: 0,
  250. more_frags: true,
  251. ident: 12345,
  252. };
  253. let mut bytes = [0u8; 8];
  254. let mut header = Header::new_unchecked(&mut bytes);
  255. repr.emit(&mut header);
  256. assert_eq!(header.into_inner(), &BYTES_HEADER_MORE_FRAG[0..8]);
  257. let repr = Repr {
  258. next_header: Protocol::Tcp,
  259. frag_offset: 320,
  260. more_frags: false,
  261. ident: 67890,
  262. };
  263. let mut bytes = [0u8; 8];
  264. let mut header = Header::new_unchecked(&mut bytes);
  265. repr.emit(&mut header);
  266. assert_eq!(header.into_inner(), &BYTES_HEADER_LAST_FRAG[0..8]);
  267. }
  268. #[test]
  269. fn test_buffer_len() {
  270. let header = Header::new_unchecked(&BYTES_HEADER_MORE_FRAG);
  271. let repr = Repr::parse(&header).unwrap();
  272. assert_eq!(repr.buffer_len(), BYTES_HEADER_MORE_FRAG.len());
  273. }
  274. }