ethernet.rs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385
  1. use core::fmt;
  2. use byteorder::{ByteOrder, NetworkEndian};
  3. use crate::{Error, Result};
  4. enum_with_unknown! {
  5. /// Ethernet protocol type.
  6. pub enum EtherType(u16) {
  7. Ipv4 = 0x0800,
  8. Arp = 0x0806,
  9. Ipv6 = 0x86DD
  10. }
  11. }
  12. impl fmt::Display for EtherType {
  13. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  14. match *self {
  15. EtherType::Ipv4 => write!(f, "IPv4"),
  16. EtherType::Ipv6 => write!(f, "IPv6"),
  17. EtherType::Arp => write!(f, "ARP"),
  18. EtherType::Unknown(id) => write!(f, "0x{:04x}", id)
  19. }
  20. }
  21. }
  22. /// A six-octet Ethernet II address.
  23. #[derive(Debug, Hash, PartialEq, Eq, PartialOrd, Ord, Clone, Copy, Default)]
  24. pub struct Address(pub [u8; 6]);
  25. impl Address {
  26. /// The broadcast address.
  27. pub const BROADCAST: Address = Address([0xff; 6]);
  28. /// Construct an Ethernet address from a sequence of octets, in big-endian.
  29. ///
  30. /// # Panics
  31. /// The function panics if `data` is not six octets long.
  32. pub fn from_bytes(data: &[u8]) -> Address {
  33. let mut bytes = [0; 6];
  34. bytes.copy_from_slice(data);
  35. Address(bytes)
  36. }
  37. /// Return an Ethernet address as a sequence of octets, in big-endian.
  38. pub fn as_bytes(&self) -> &[u8] {
  39. &self.0
  40. }
  41. /// Query whether the address is an unicast address.
  42. pub fn is_unicast(&self) -> bool {
  43. !(self.is_broadcast() ||
  44. self.is_multicast())
  45. }
  46. /// Query whether this address is the broadcast address.
  47. pub fn is_broadcast(&self) -> bool {
  48. *self == Self::BROADCAST
  49. }
  50. /// Query whether the "multicast" bit in the OUI is set.
  51. pub fn is_multicast(&self) -> bool {
  52. self.0[0] & 0x01 != 0
  53. }
  54. /// Query whether the "locally administered" bit in the OUI is set.
  55. pub fn is_local(&self) -> bool {
  56. self.0[0] & 0x02 != 0
  57. }
  58. }
  59. impl fmt::Display for Address {
  60. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  61. let bytes = self.0;
  62. write!(f, "{:02x}-{:02x}-{:02x}-{:02x}-{:02x}-{:02x}",
  63. bytes[0], bytes[1], bytes[2], bytes[3], bytes[4], bytes[5])
  64. }
  65. }
  66. /// A read/write wrapper around an Ethernet II frame buffer.
  67. #[derive(Debug, Clone)]
  68. pub struct Frame<T: AsRef<[u8]>> {
  69. buffer: T
  70. }
  71. mod field {
  72. use crate::wire::field::*;
  73. pub const DESTINATION: Field = 0..6;
  74. pub const SOURCE: Field = 6..12;
  75. pub const ETHERTYPE: Field = 12..14;
  76. pub const PAYLOAD: Rest = 14..;
  77. }
  78. /// The Ethernet header length
  79. pub const HEADER_LEN: usize = field::PAYLOAD.start;
  80. impl<T: AsRef<[u8]>> Frame<T> {
  81. /// Imbue a raw octet buffer with Ethernet frame structure.
  82. pub fn new_unchecked(buffer: T) -> Frame<T> {
  83. Frame { buffer }
  84. }
  85. /// Shorthand for a combination of [new_unchecked] and [check_len].
  86. ///
  87. /// [new_unchecked]: #method.new_unchecked
  88. /// [check_len]: #method.check_len
  89. pub fn new_checked(buffer: T) -> Result<Frame<T>> {
  90. let packet = Self::new_unchecked(buffer);
  91. packet.check_len()?;
  92. Ok(packet)
  93. }
  94. /// Ensure that no accessor method will panic if called.
  95. /// Returns `Err(Error::Truncated)` if the buffer is too short.
  96. pub fn check_len(&self) -> Result<()> {
  97. let len = self.buffer.as_ref().len();
  98. if len < HEADER_LEN {
  99. Err(Error::Truncated)
  100. } else {
  101. Ok(())
  102. }
  103. }
  104. /// Consumes the frame, returning the underlying buffer.
  105. pub fn into_inner(self) -> T {
  106. self.buffer
  107. }
  108. /// Return the length of a frame header.
  109. pub fn header_len() -> usize {
  110. HEADER_LEN
  111. }
  112. /// Return the length of a buffer required to hold a packet with the payload
  113. /// of a given length.
  114. pub fn buffer_len(payload_len: usize) -> usize {
  115. HEADER_LEN + payload_len
  116. }
  117. /// Return the destination address field.
  118. #[inline]
  119. pub fn dst_addr(&self) -> Address {
  120. let data = self.buffer.as_ref();
  121. Address::from_bytes(&data[field::DESTINATION])
  122. }
  123. /// Return the source address field.
  124. #[inline]
  125. pub fn src_addr(&self) -> Address {
  126. let data = self.buffer.as_ref();
  127. Address::from_bytes(&data[field::SOURCE])
  128. }
  129. /// Return the EtherType field, without checking for 802.1Q.
  130. #[inline]
  131. pub fn ethertype(&self) -> EtherType {
  132. let data = self.buffer.as_ref();
  133. let raw = NetworkEndian::read_u16(&data[field::ETHERTYPE]);
  134. EtherType::from(raw)
  135. }
  136. }
  137. impl<'a, T: AsRef<[u8]> + ?Sized> Frame<&'a T> {
  138. /// Return a pointer to the payload, without checking for 802.1Q.
  139. #[inline]
  140. pub fn payload(&self) -> &'a [u8] {
  141. let data = self.buffer.as_ref();
  142. &data[field::PAYLOAD]
  143. }
  144. }
  145. impl<T: AsRef<[u8]> + AsMut<[u8]>> Frame<T> {
  146. /// Set the destination address field.
  147. #[inline]
  148. pub fn set_dst_addr(&mut self, value: Address) {
  149. let data = self.buffer.as_mut();
  150. data[field::DESTINATION].copy_from_slice(value.as_bytes())
  151. }
  152. /// Set the source address field.
  153. #[inline]
  154. pub fn set_src_addr(&mut self, value: Address) {
  155. let data = self.buffer.as_mut();
  156. data[field::SOURCE].copy_from_slice(value.as_bytes())
  157. }
  158. /// Set the EtherType field.
  159. #[inline]
  160. pub fn set_ethertype(&mut self, value: EtherType) {
  161. let data = self.buffer.as_mut();
  162. NetworkEndian::write_u16(&mut data[field::ETHERTYPE], value.into())
  163. }
  164. /// Return a mutable pointer to the payload.
  165. #[inline]
  166. pub fn payload_mut(&mut self) -> &mut [u8] {
  167. let data = self.buffer.as_mut();
  168. &mut data[field::PAYLOAD]
  169. }
  170. }
  171. impl<T: AsRef<[u8]>> AsRef<[u8]> for Frame<T> {
  172. fn as_ref(&self) -> &[u8] {
  173. self.buffer.as_ref()
  174. }
  175. }
  176. impl<T: AsRef<[u8]>> fmt::Display for Frame<T> {
  177. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  178. write!(f, "EthernetII src={} dst={} type={}",
  179. self.src_addr(), self.dst_addr(), self.ethertype())
  180. }
  181. }
  182. use crate::wire::pretty_print::{PrettyPrint, PrettyIndent};
  183. impl<T: AsRef<[u8]>> PrettyPrint for Frame<T> {
  184. fn pretty_print(buffer: &dyn AsRef<[u8]>, f: &mut fmt::Formatter,
  185. indent: &mut PrettyIndent) -> fmt::Result {
  186. let frame = match Frame::new_checked(buffer) {
  187. Err(err) => return write!(f, "{}({})", indent, err),
  188. Ok(frame) => frame
  189. };
  190. write!(f, "{}{}", indent, frame)?;
  191. match frame.ethertype() {
  192. #[cfg(feature = "proto-ipv4")]
  193. EtherType::Arp => {
  194. indent.increase(f)?;
  195. super::ArpPacket::<&[u8]>::pretty_print(&frame.payload(), f, indent)
  196. }
  197. #[cfg(feature = "proto-ipv4")]
  198. EtherType::Ipv4 => {
  199. indent.increase(f)?;
  200. super::Ipv4Packet::<&[u8]>::pretty_print(&frame.payload(), f, indent)
  201. }
  202. #[cfg(feature = "proto-ipv6")]
  203. EtherType::Ipv6 => {
  204. indent.increase(f)?;
  205. super::Ipv6Packet::<&[u8]>::pretty_print(&frame.payload(), f, indent)
  206. }
  207. _ => Ok(())
  208. }
  209. }
  210. }
  211. /// A high-level representation of an Internet Protocol version 4 packet header.
  212. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  213. pub struct Repr {
  214. pub src_addr: Address,
  215. pub dst_addr: Address,
  216. pub ethertype: EtherType,
  217. }
  218. impl Repr {
  219. /// Parse an Ethernet II frame and return a high-level representation.
  220. pub fn parse<T: AsRef<[u8]> + ?Sized>(frame: &Frame<&T>) -> Result<Repr> {
  221. frame.check_len()?;
  222. Ok(Repr {
  223. src_addr: frame.src_addr(),
  224. dst_addr: frame.dst_addr(),
  225. ethertype: frame.ethertype(),
  226. })
  227. }
  228. /// Return the length of a header that will be emitted from this high-level representation.
  229. pub fn buffer_len(&self) -> usize {
  230. HEADER_LEN
  231. }
  232. /// Emit a high-level representation into an Ethernet II frame.
  233. pub fn emit<T: AsRef<[u8]> + AsMut<[u8]>>(&self, frame: &mut Frame<T>) {
  234. frame.set_src_addr(self.src_addr);
  235. frame.set_dst_addr(self.dst_addr);
  236. frame.set_ethertype(self.ethertype);
  237. }
  238. }
  239. #[cfg(test)]
  240. mod test {
  241. // Tests that are valid with any combination of
  242. // "proto-*" features.
  243. use super::*;
  244. #[test]
  245. fn test_broadcast() {
  246. assert!(Address::BROADCAST.is_broadcast());
  247. assert!(!Address::BROADCAST.is_unicast());
  248. assert!(Address::BROADCAST.is_multicast());
  249. assert!(Address::BROADCAST.is_local());
  250. }
  251. }
  252. #[cfg(test)]
  253. #[cfg(feature = "proto-ipv4")]
  254. mod test_ipv4 {
  255. // Tests that are valid only with "proto-ipv4"
  256. use super::*;
  257. static FRAME_BYTES: [u8; 64] =
  258. [0x01, 0x02, 0x03, 0x04, 0x05, 0x06,
  259. 0x11, 0x12, 0x13, 0x14, 0x15, 0x16,
  260. 0x08, 0x00,
  261. 0xaa, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  262. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  263. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  264. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  265. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  266. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  267. 0x00, 0xff];
  268. static PAYLOAD_BYTES: [u8; 50] =
  269. [0xaa, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  270. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  271. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  272. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  273. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  274. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  275. 0x00, 0xff];
  276. #[test]
  277. fn test_deconstruct() {
  278. let frame = Frame::new_unchecked(&FRAME_BYTES[..]);
  279. assert_eq!(frame.dst_addr(), Address([0x01, 0x02, 0x03, 0x04, 0x05, 0x06]));
  280. assert_eq!(frame.src_addr(), Address([0x11, 0x12, 0x13, 0x14, 0x15, 0x16]));
  281. assert_eq!(frame.ethertype(), EtherType::Ipv4);
  282. assert_eq!(frame.payload(), &PAYLOAD_BYTES[..]);
  283. }
  284. #[test]
  285. fn test_construct() {
  286. let mut bytes = vec![0xa5; 64];
  287. let mut frame = Frame::new_unchecked(&mut bytes);
  288. frame.set_dst_addr(Address([0x01, 0x02, 0x03, 0x04, 0x05, 0x06]));
  289. frame.set_src_addr(Address([0x11, 0x12, 0x13, 0x14, 0x15, 0x16]));
  290. frame.set_ethertype(EtherType::Ipv4);
  291. frame.payload_mut().copy_from_slice(&PAYLOAD_BYTES[..]);
  292. assert_eq!(&frame.into_inner()[..], &FRAME_BYTES[..]);
  293. }
  294. }
  295. #[cfg(test)]
  296. #[cfg(feature = "proto-ipv6")]
  297. mod test_ipv6 {
  298. // Tests that are valid only with "proto-ipv6"
  299. use super::*;
  300. static FRAME_BYTES: [u8; 54] =
  301. [0x01, 0x02, 0x03, 0x04, 0x05, 0x06,
  302. 0x11, 0x12, 0x13, 0x14, 0x15, 0x16,
  303. 0x86, 0xdd,
  304. 0x60, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  305. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  306. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
  307. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  308. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01];
  309. static PAYLOAD_BYTES: [u8; 40] =
  310. [0x60, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  311. 0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  312. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
  313. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  314. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01];
  315. #[test]
  316. fn test_deconstruct() {
  317. let frame = Frame::new_unchecked(&FRAME_BYTES[..]);
  318. assert_eq!(frame.dst_addr(), Address([0x01, 0x02, 0x03, 0x04, 0x05, 0x06]));
  319. assert_eq!(frame.src_addr(), Address([0x11, 0x12, 0x13, 0x14, 0x15, 0x16]));
  320. assert_eq!(frame.ethertype(), EtherType::Ipv6);
  321. assert_eq!(frame.payload(), &PAYLOAD_BYTES[..]);
  322. }
  323. #[test]
  324. fn test_construct() {
  325. let mut bytes = vec![0xa5; 54];
  326. let mut frame = Frame::new_unchecked(&mut bytes);
  327. frame.set_dst_addr(Address([0x01, 0x02, 0x03, 0x04, 0x05, 0x06]));
  328. frame.set_src_addr(Address([0x11, 0x12, 0x13, 0x14, 0x15, 0x16]));
  329. frame.set_ethertype(EtherType::Ipv6);
  330. assert_eq!(PAYLOAD_BYTES.len(), frame.payload_mut().len());
  331. frame.payload_mut().copy_from_slice(&PAYLOAD_BYTES[..]);
  332. assert_eq!(&frame.into_inner()[..], &FRAME_BYTES[..]);
  333. }
  334. }