ipv6hbh.rs 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. use super::{Error, Ipv6Option, Ipv6OptionRepr, Ipv6OptionsIterator, Result};
  2. use heapless::Vec;
  3. /// A read/write wrapper around an IPv6 Hop-by-Hop Header buffer.
  4. pub struct Header<T: AsRef<[u8]>> {
  5. buffer: T,
  6. }
  7. impl<T: AsRef<[u8]>> Header<T> {
  8. /// Create a raw octet buffer with an IPv6 Hop-by-Hop Header structure.
  9. pub const fn new_unchecked(buffer: T) -> Self {
  10. Header { buffer }
  11. }
  12. /// Shorthand for a combination of [new_unchecked] and [check_len].
  13. ///
  14. /// [new_unchecked]: #method.new_unchecked
  15. /// [check_len]: #method.check_len
  16. pub fn new_checked(buffer: T) -> Result<Self> {
  17. let header = Self::new_unchecked(buffer);
  18. header.check_len()?;
  19. Ok(header)
  20. }
  21. /// Ensure that no accessor method will panic if called.
  22. /// Returns `Err(Error)` if the buffer is too short.
  23. ///
  24. /// The result of this check is invalidated by calling [set_header_len].
  25. ///
  26. /// [set_header_len]: #method.set_header_len
  27. pub fn check_len(&self) -> Result<()> {
  28. if self.buffer.as_ref().is_empty() {
  29. return Err(Error);
  30. }
  31. Ok(())
  32. }
  33. /// Consume the header, returning the underlying buffer.
  34. pub fn into_inner(self) -> T {
  35. self.buffer
  36. }
  37. }
  38. impl<'a, T: AsRef<[u8]> + ?Sized> Header<&'a T> {
  39. /// Return the options of the IPv6 Hop-by-Hop header.
  40. pub fn options(&self) -> &'a [u8] {
  41. self.buffer.as_ref()
  42. }
  43. }
  44. impl<'a, T: AsRef<[u8]> + AsMut<[u8]> + ?Sized> Header<&'a mut T> {
  45. /// Return a mutable pointer to the options of the IPv6 Hop-by-Hop header.
  46. pub fn options_mut(&mut self) -> &mut [u8] {
  47. self.buffer.as_mut()
  48. }
  49. }
  50. /// A high-level representation of an IPv6 Hop-by-Hop Header.
  51. #[derive(Debug, PartialEq, Eq, Clone)]
  52. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  53. pub struct Repr<'a> {
  54. pub options: heapless::Vec<Ipv6OptionRepr<'a>, { crate::config::IPV6_HBH_MAX_OPTIONS }>,
  55. }
  56. impl<'a> Repr<'a> {
  57. /// Parse an IPv6 Hop-by-Hop Header and return a high-level representation.
  58. pub fn parse<T>(header: &'a Header<&'a T>) -> Result<Repr<'a>>
  59. where
  60. T: AsRef<[u8]> + ?Sized,
  61. {
  62. let mut options = Vec::new();
  63. let iter = Ipv6OptionsIterator::new(header.options());
  64. for option in iter {
  65. let option = option?;
  66. if let Err(e) = options.push(option) {
  67. net_trace!("eror when parsing hop-by-hop options: {}", e);
  68. break;
  69. }
  70. }
  71. Ok(Self { options })
  72. }
  73. /// Return the length, in bytes, of a header that will be emitted from this high-level
  74. /// representation.
  75. pub fn buffer_len(&self) -> usize {
  76. self.options.iter().map(|o| o.buffer_len()).sum()
  77. }
  78. /// Emit a high-level representation into an IPv6 Hop-by-Hop Header.
  79. pub fn emit<T: AsRef<[u8]> + AsMut<[u8]> + ?Sized>(&self, header: &mut Header<&mut T>) {
  80. let mut buffer = header.options_mut();
  81. for opt in &self.options {
  82. opt.emit(&mut Ipv6Option::new_unchecked(
  83. &mut buffer[..opt.buffer_len()],
  84. ));
  85. buffer = &mut buffer[opt.buffer_len()..];
  86. }
  87. }
  88. }
  89. #[cfg(test)]
  90. mod tests {
  91. use super::*;
  92. use crate::wire::Error;
  93. // A Hop-by-Hop Option header with a PadN option of option data length 4.
  94. static REPR_PACKET_PAD4: [u8; 6] = [0x1, 0x4, 0x0, 0x0, 0x0, 0x0];
  95. // A Hop-by-Hop Option header with a PadN option of option data length 12.
  96. static REPR_PACKET_PAD12: [u8; 14] = [
  97. 0x1, 0x0C, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  98. ];
  99. #[test]
  100. fn test_check_len() {
  101. // zero byte buffer
  102. assert_eq!(
  103. Err(Error),
  104. Header::new_unchecked(&REPR_PACKET_PAD4[..0]).check_len()
  105. );
  106. // valid
  107. assert_eq!(Ok(()), Header::new_unchecked(&REPR_PACKET_PAD4).check_len());
  108. // valid
  109. assert_eq!(
  110. Ok(()),
  111. Header::new_unchecked(&REPR_PACKET_PAD12).check_len()
  112. );
  113. }
  114. #[test]
  115. fn test_repr_parse_valid() {
  116. let header = Header::new_unchecked(&REPR_PACKET_PAD4);
  117. let repr = Repr::parse(&header).unwrap();
  118. let mut options = Vec::new();
  119. options.push(Ipv6OptionRepr::PadN(4)).unwrap();
  120. assert_eq!(repr, Repr { options });
  121. let header = Header::new_unchecked(&REPR_PACKET_PAD12);
  122. let repr = Repr::parse(&header).unwrap();
  123. let mut options = Vec::new();
  124. options.push(Ipv6OptionRepr::PadN(12)).unwrap();
  125. assert_eq!(repr, Repr { options });
  126. }
  127. #[test]
  128. fn test_repr_emit() {
  129. let mut options = Vec::new();
  130. options.push(Ipv6OptionRepr::PadN(4)).unwrap();
  131. let repr = Repr { options };
  132. let mut bytes = [0u8; 6];
  133. let mut header = Header::new_unchecked(&mut bytes);
  134. repr.emit(&mut header);
  135. assert_eq!(header.into_inner(), &REPR_PACKET_PAD4[..]);
  136. let mut options = Vec::new();
  137. options.push(Ipv6OptionRepr::PadN(12)).unwrap();
  138. let repr = Repr { options };
  139. let mut bytes = [0u8; 14];
  140. let mut header = Header::new_unchecked(&mut bytes);
  141. repr.emit(&mut header);
  142. assert_eq!(header.into_inner(), &REPR_PACKET_PAD12[..]);
  143. }
  144. }