arp.rs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444
  1. use core::fmt;
  2. use byteorder::{ByteOrder, NetworkEndian};
  3. use {Error, Result};
  4. pub use super::EthernetProtocol as Protocol;
  5. enum_with_unknown! {
  6. /// ARP hardware type.
  7. pub enum Hardware(u16) {
  8. Ethernet = 1
  9. }
  10. }
  11. enum_with_unknown! {
  12. /// ARP operation type.
  13. pub enum Operation(u16) {
  14. Request = 1,
  15. Reply = 2
  16. }
  17. }
  18. /// A read/write wrapper around an Address Resolution Protocol packet buffer.
  19. #[derive(Debug, PartialEq, Clone)]
  20. pub struct Packet<T: AsRef<[u8]>> {
  21. buffer: T
  22. }
  23. mod field {
  24. #![allow(non_snake_case)]
  25. use wire::field::*;
  26. pub const HTYPE: Field = 0..2;
  27. pub const PTYPE: Field = 2..4;
  28. pub const HLEN: usize = 4;
  29. pub const PLEN: usize = 5;
  30. pub const OPER: Field = 6..8;
  31. #[inline]
  32. pub fn SHA(hardware_len: u8, _protocol_len: u8) -> Field {
  33. let start = OPER.end;
  34. start..(start + hardware_len as usize)
  35. }
  36. #[inline]
  37. pub fn SPA(hardware_len: u8, protocol_len: u8) -> Field {
  38. let start = SHA(hardware_len, protocol_len).end;
  39. start..(start + protocol_len as usize)
  40. }
  41. #[inline]
  42. pub fn THA(hardware_len: u8, protocol_len: u8) -> Field {
  43. let start = SPA(hardware_len, protocol_len).end;
  44. start..(start + hardware_len as usize)
  45. }
  46. #[inline]
  47. pub fn TPA(hardware_len: u8, protocol_len: u8) -> Field {
  48. let start = THA(hardware_len, protocol_len).end;
  49. start..(start + protocol_len as usize)
  50. }
  51. }
  52. impl<T: AsRef<[u8]>> Packet<T> {
  53. /// Imbue a raw octet buffer with ARP packet structure.
  54. pub fn new_unchecked(buffer: T) -> Packet<T> {
  55. Packet { buffer }
  56. }
  57. /// Shorthand for a combination of [new_unchecked] and [check_len].
  58. ///
  59. /// [new_unchecked]: #method.new_unchecked
  60. /// [check_len]: #method.check_len
  61. pub fn new_checked(buffer: T) -> Result<Packet<T>> {
  62. let packet = Self::new_unchecked(buffer);
  63. packet.check_len()?;
  64. Ok(packet)
  65. }
  66. /// Ensure that no accessor method will panic if called.
  67. /// Returns `Err(Error::Truncated)` if the buffer is too short.
  68. ///
  69. /// The result of this check is invalidated by calling [set_hardware_len] or
  70. /// [set_protocol_len].
  71. ///
  72. /// [set_hardware_len]: #method.set_hardware_len
  73. /// [set_protocol_len]: #method.set_protocol_len
  74. pub fn check_len(&self) -> Result<()> {
  75. let len = self.buffer.as_ref().len();
  76. if len < field::OPER.end {
  77. Err(Error::Truncated)
  78. } else if len < field::TPA(self.hardware_len(), self.protocol_len()).end {
  79. Err(Error::Truncated)
  80. } else {
  81. Ok(())
  82. }
  83. }
  84. /// Consume the packet, returning the underlying buffer.
  85. pub fn into_inner(self) -> T {
  86. self.buffer
  87. }
  88. /// Return the hardware type field.
  89. #[inline]
  90. pub fn hardware_type(&self) -> Hardware {
  91. let data = self.buffer.as_ref();
  92. let raw = NetworkEndian::read_u16(&data[field::HTYPE]);
  93. Hardware::from(raw)
  94. }
  95. /// Return the protocol type field.
  96. #[inline]
  97. pub fn protocol_type(&self) -> Protocol {
  98. let data = self.buffer.as_ref();
  99. let raw = NetworkEndian::read_u16(&data[field::PTYPE]);
  100. Protocol::from(raw)
  101. }
  102. /// Return the hardware length field.
  103. #[inline]
  104. pub fn hardware_len(&self) -> u8 {
  105. let data = self.buffer.as_ref();
  106. data[field::HLEN]
  107. }
  108. /// Return the protocol length field.
  109. #[inline]
  110. pub fn protocol_len(&self) -> u8 {
  111. let data = self.buffer.as_ref();
  112. data[field::PLEN]
  113. }
  114. /// Return the operation field.
  115. #[inline]
  116. pub fn operation(&self) -> Operation {
  117. let data = self.buffer.as_ref();
  118. let raw = NetworkEndian::read_u16(&data[field::OPER]);
  119. Operation::from(raw)
  120. }
  121. /// Return the source hardware address field.
  122. pub fn source_hardware_addr(&self) -> &[u8] {
  123. let data = self.buffer.as_ref();
  124. &data[field::SHA(self.hardware_len(), self.protocol_len())]
  125. }
  126. /// Return the source protocol address field.
  127. pub fn source_protocol_addr(&self) -> &[u8] {
  128. let data = self.buffer.as_ref();
  129. &data[field::SPA(self.hardware_len(), self.protocol_len())]
  130. }
  131. /// Return the target hardware address field.
  132. pub fn target_hardware_addr(&self) -> &[u8] {
  133. let data = self.buffer.as_ref();
  134. &data[field::THA(self.hardware_len(), self.protocol_len())]
  135. }
  136. /// Return the target protocol address field.
  137. pub fn target_protocol_addr(&self) -> &[u8] {
  138. let data = self.buffer.as_ref();
  139. &data[field::TPA(self.hardware_len(), self.protocol_len())]
  140. }
  141. }
  142. impl<T: AsRef<[u8]> + AsMut<[u8]>> Packet<T> {
  143. /// Set the hardware type field.
  144. #[inline]
  145. pub fn set_hardware_type(&mut self, value: Hardware) {
  146. let data = self.buffer.as_mut();
  147. NetworkEndian::write_u16(&mut data[field::HTYPE], value.into())
  148. }
  149. /// Set the protocol type field.
  150. #[inline]
  151. pub fn set_protocol_type(&mut self, value: Protocol) {
  152. let data = self.buffer.as_mut();
  153. NetworkEndian::write_u16(&mut data[field::PTYPE], value.into())
  154. }
  155. /// Set the hardware length field.
  156. #[inline]
  157. pub fn set_hardware_len(&mut self, value: u8) {
  158. let data = self.buffer.as_mut();
  159. data[field::HLEN] = value
  160. }
  161. /// Set the protocol length field.
  162. #[inline]
  163. pub fn set_protocol_len(&mut self, value: u8) {
  164. let data = self.buffer.as_mut();
  165. data[field::PLEN] = value
  166. }
  167. /// Set the operation field.
  168. #[inline]
  169. pub fn set_operation(&mut self, value: Operation) {
  170. let data = self.buffer.as_mut();
  171. NetworkEndian::write_u16(&mut data[field::OPER], value.into())
  172. }
  173. /// Set the source hardware address field.
  174. ///
  175. /// # Panics
  176. /// The function panics if `value` is not `self.hardware_len()` long.
  177. pub fn set_source_hardware_addr(&mut self, value: &[u8]) {
  178. let (hardware_len, protocol_len) = (self.hardware_len(), self.protocol_len());
  179. let data = self.buffer.as_mut();
  180. data[field::SHA(hardware_len, protocol_len)].copy_from_slice(value)
  181. }
  182. /// Set the source protocol address field.
  183. ///
  184. /// # Panics
  185. /// The function panics if `value` is not `self.protocol_len()` long.
  186. pub fn set_source_protocol_addr(&mut self, value: &[u8]) {
  187. let (hardware_len, protocol_len) = (self.hardware_len(), self.protocol_len());
  188. let data = self.buffer.as_mut();
  189. data[field::SPA(hardware_len, protocol_len)].copy_from_slice(value)
  190. }
  191. /// Set the target hardware address field.
  192. ///
  193. /// # Panics
  194. /// The function panics if `value` is not `self.hardware_len()` long.
  195. pub fn set_target_hardware_addr(&mut self, value: &[u8]) {
  196. let (hardware_len, protocol_len) = (self.hardware_len(), self.protocol_len());
  197. let data = self.buffer.as_mut();
  198. data[field::THA(hardware_len, protocol_len)].copy_from_slice(value)
  199. }
  200. /// Set the target protocol address field.
  201. ///
  202. /// # Panics
  203. /// The function panics if `value` is not `self.protocol_len()` long.
  204. pub fn set_target_protocol_addr(&mut self, value: &[u8]) {
  205. let (hardware_len, protocol_len) = (self.hardware_len(), self.protocol_len());
  206. let data = self.buffer.as_mut();
  207. data[field::TPA(hardware_len, protocol_len)].copy_from_slice(value)
  208. }
  209. }
  210. impl<T: AsRef<[u8]>> AsRef<[u8]> for Packet<T> {
  211. fn as_ref(&self) -> &[u8] {
  212. self.buffer.as_ref()
  213. }
  214. }
  215. use super::{EthernetAddress, Ipv4Address};
  216. /// A high-level representation of an Address Resolution Protocol packet.
  217. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  218. pub enum Repr {
  219. /// An Ethernet and IPv4 Address Resolution Protocol packet.
  220. EthernetIpv4 {
  221. operation: Operation,
  222. source_hardware_addr: EthernetAddress,
  223. source_protocol_addr: Ipv4Address,
  224. target_hardware_addr: EthernetAddress,
  225. target_protocol_addr: Ipv4Address
  226. },
  227. #[doc(hidden)]
  228. __Nonexhaustive
  229. }
  230. impl Repr {
  231. /// Parse an Address Resolution Protocol packet and return a high-level representation,
  232. /// or return `Err(Error::Unrecognized)` if the packet is not recognized.
  233. pub fn parse<T: AsRef<[u8]>>(packet: &Packet<T>) -> Result<Repr> {
  234. match (packet.hardware_type(), packet.protocol_type(),
  235. packet.hardware_len(), packet.protocol_len()) {
  236. (Hardware::Ethernet, Protocol::Ipv4, 6, 4) => {
  237. Ok(Repr::EthernetIpv4 {
  238. operation: packet.operation(),
  239. source_hardware_addr:
  240. EthernetAddress::from_bytes(packet.source_hardware_addr()),
  241. source_protocol_addr:
  242. Ipv4Address::from_bytes(packet.source_protocol_addr()),
  243. target_hardware_addr:
  244. EthernetAddress::from_bytes(packet.target_hardware_addr()),
  245. target_protocol_addr:
  246. Ipv4Address::from_bytes(packet.target_protocol_addr())
  247. })
  248. },
  249. _ => Err(Error::Unrecognized)
  250. }
  251. }
  252. /// Return the length of a packet that will be emitted from this high-level representation.
  253. pub fn buffer_len(&self) -> usize {
  254. match *self {
  255. Repr::EthernetIpv4 { .. } => field::TPA(6, 4).end,
  256. Repr::__Nonexhaustive => unreachable!()
  257. }
  258. }
  259. /// Emit a high-level representation into an Address Resolution Protocol packet.
  260. pub fn emit<T: AsRef<[u8]> + AsMut<[u8]>>(&self, packet: &mut Packet<T>) {
  261. match *self {
  262. Repr::EthernetIpv4 {
  263. operation,
  264. source_hardware_addr, source_protocol_addr,
  265. target_hardware_addr, target_protocol_addr
  266. } => {
  267. packet.set_hardware_type(Hardware::Ethernet);
  268. packet.set_protocol_type(Protocol::Ipv4);
  269. packet.set_hardware_len(6);
  270. packet.set_protocol_len(4);
  271. packet.set_operation(operation);
  272. packet.set_source_hardware_addr(source_hardware_addr.as_bytes());
  273. packet.set_source_protocol_addr(source_protocol_addr.as_bytes());
  274. packet.set_target_hardware_addr(target_hardware_addr.as_bytes());
  275. packet.set_target_protocol_addr(target_protocol_addr.as_bytes());
  276. },
  277. Repr::__Nonexhaustive => unreachable!()
  278. }
  279. }
  280. }
  281. impl<T: AsRef<[u8]>> fmt::Display for Packet<T> {
  282. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  283. match Repr::parse(self) {
  284. Ok(repr) => write!(f, "{}", repr),
  285. _ => {
  286. write!(f, "ARP (unrecognized)")?;
  287. write!(f, " htype={:?} ptype={:?} hlen={:?} plen={:?} op={:?}",
  288. self.hardware_type(), self.protocol_type(),
  289. self.hardware_len(), self.protocol_len(),
  290. self.operation())?;
  291. write!(f, " sha={:?} spa={:?} tha={:?} tpa={:?}",
  292. self.source_hardware_addr(), self.source_protocol_addr(),
  293. self.target_hardware_addr(), self.target_protocol_addr())?;
  294. Ok(())
  295. }
  296. }
  297. }
  298. }
  299. impl fmt::Display for Repr {
  300. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  301. match *self {
  302. Repr::EthernetIpv4 {
  303. operation,
  304. source_hardware_addr, source_protocol_addr,
  305. target_hardware_addr, target_protocol_addr
  306. } => {
  307. write!(f, "ARP type=Ethernet+IPv4 src={}/{} tgt={}/{} op={:?}",
  308. source_hardware_addr, source_protocol_addr,
  309. target_hardware_addr, target_protocol_addr,
  310. operation)
  311. },
  312. Repr::__Nonexhaustive => unreachable!()
  313. }
  314. }
  315. }
  316. use super::pretty_print::{PrettyPrint, PrettyIndent};
  317. impl<T: AsRef<[u8]>> PrettyPrint for Packet<T> {
  318. fn pretty_print(buffer: &dyn AsRef<[u8]>, f: &mut fmt::Formatter,
  319. indent: &mut PrettyIndent) -> fmt::Result {
  320. match Packet::new_checked(buffer) {
  321. Err(err) => write!(f, "{}({})", indent, err),
  322. Ok(packet) => write!(f, "{}{}", indent, packet)
  323. }
  324. }
  325. }
  326. #[cfg(test)]
  327. mod test {
  328. use super::*;
  329. static PACKET_BYTES: [u8; 28] =
  330. [0x00, 0x01,
  331. 0x08, 0x00,
  332. 0x06,
  333. 0x04,
  334. 0x00, 0x01,
  335. 0x11, 0x12, 0x13, 0x14, 0x15, 0x16,
  336. 0x21, 0x22, 0x23, 0x24,
  337. 0x31, 0x32, 0x33, 0x34, 0x35, 0x36,
  338. 0x41, 0x42, 0x43, 0x44];
  339. #[test]
  340. fn test_deconstruct() {
  341. let packet = Packet::new_unchecked(&PACKET_BYTES[..]);
  342. assert_eq!(packet.hardware_type(), Hardware::Ethernet);
  343. assert_eq!(packet.protocol_type(), Protocol::Ipv4);
  344. assert_eq!(packet.hardware_len(), 6);
  345. assert_eq!(packet.protocol_len(), 4);
  346. assert_eq!(packet.operation(), Operation::Request);
  347. assert_eq!(packet.source_hardware_addr(), &[0x11, 0x12, 0x13, 0x14, 0x15, 0x16]);
  348. assert_eq!(packet.source_protocol_addr(), &[0x21, 0x22, 0x23, 0x24]);
  349. assert_eq!(packet.target_hardware_addr(), &[0x31, 0x32, 0x33, 0x34, 0x35, 0x36]);
  350. assert_eq!(packet.target_protocol_addr(), &[0x41, 0x42, 0x43, 0x44]);
  351. }
  352. #[test]
  353. fn test_construct() {
  354. let mut bytes = vec![0xa5; 28];
  355. let mut packet = Packet::new_unchecked(&mut bytes);
  356. packet.set_hardware_type(Hardware::Ethernet);
  357. packet.set_protocol_type(Protocol::Ipv4);
  358. packet.set_hardware_len(6);
  359. packet.set_protocol_len(4);
  360. packet.set_operation(Operation::Request);
  361. packet.set_source_hardware_addr(&[0x11, 0x12, 0x13, 0x14, 0x15, 0x16]);
  362. packet.set_source_protocol_addr(&[0x21, 0x22, 0x23, 0x24]);
  363. packet.set_target_hardware_addr(&[0x31, 0x32, 0x33, 0x34, 0x35, 0x36]);
  364. packet.set_target_protocol_addr(&[0x41, 0x42, 0x43, 0x44]);
  365. assert_eq!(&packet.into_inner()[..], &PACKET_BYTES[..]);
  366. }
  367. fn packet_repr() -> Repr {
  368. Repr::EthernetIpv4 {
  369. operation: Operation::Request,
  370. source_hardware_addr:
  371. EthernetAddress::from_bytes(&[0x11, 0x12, 0x13, 0x14, 0x15, 0x16]),
  372. source_protocol_addr:
  373. Ipv4Address::from_bytes(&[0x21, 0x22, 0x23, 0x24]),
  374. target_hardware_addr:
  375. EthernetAddress::from_bytes(&[0x31, 0x32, 0x33, 0x34, 0x35, 0x36]),
  376. target_protocol_addr:
  377. Ipv4Address::from_bytes(&[0x41, 0x42, 0x43, 0x44])
  378. }
  379. }
  380. #[test]
  381. fn test_parse() {
  382. let packet = Packet::new_unchecked(&PACKET_BYTES[..]);
  383. let repr = Repr::parse(&packet).unwrap();
  384. assert_eq!(repr, packet_repr());
  385. }
  386. #[test]
  387. fn test_emit() {
  388. let mut bytes = vec![0xa5; 28];
  389. let mut packet = Packet::new_unchecked(&mut bytes);
  390. packet_repr().emit(&mut packet);
  391. assert_eq!(&packet.into_inner()[..], &PACKET_BYTES[..]);
  392. }
  393. }