ip.rs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537
  1. use core::fmt;
  2. use {Error, Result};
  3. use phy::ChecksumCapabilities;
  4. use super::{Ipv4Address, Ipv4Packet, Ipv4Repr};
  5. /// Internet protocol version.
  6. #[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone, Copy)]
  7. pub enum Version {
  8. Unspecified,
  9. Ipv4,
  10. #[doc(hidden)]
  11. __Nonexhaustive,
  12. }
  13. impl Version {
  14. /// Return the version of an IP packet stored in the provided buffer.
  15. ///
  16. /// This function never returns `Ok(IpVersion::Unspecified)`; instead,
  17. /// unknown versions result in `Err(Error::Unrecognized)`.
  18. pub fn of_packet(data: &[u8]) -> Result<Version> {
  19. match data[0] >> 4 {
  20. 4 => Ok(Version::Ipv4),
  21. _ => Err(Error::Unrecognized)
  22. }
  23. }
  24. }
  25. impl fmt::Display for Version {
  26. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  27. match self {
  28. &Version::Unspecified => write!(f, "IPv?"),
  29. &Version::Ipv4 => write!(f, "IPv4"),
  30. &Version::__Nonexhaustive => unreachable!()
  31. }
  32. }
  33. }
  34. enum_with_unknown! {
  35. /// IP datagram encapsulated protocol.
  36. pub enum Protocol(u8) {
  37. Icmp = 0x01,
  38. Tcp = 0x06,
  39. Udp = 0x11
  40. }
  41. }
  42. impl fmt::Display for Protocol {
  43. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  44. match self {
  45. &Protocol::Icmp => write!(f, "ICMP"),
  46. &Protocol::Tcp => write!(f, "TCP"),
  47. &Protocol::Udp => write!(f, "UDP"),
  48. &Protocol::Unknown(id) => write!(f, "0x{:02x}", id)
  49. }
  50. }
  51. }
  52. /// An internetworking address.
  53. #[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone, Copy)]
  54. pub enum Address {
  55. /// An unspecified address.
  56. /// May be used as a placeholder for storage where the address is not assigned yet.
  57. Unspecified,
  58. /// An IPv4 address.
  59. Ipv4(Ipv4Address),
  60. #[doc(hidden)]
  61. __Nonexhaustive
  62. }
  63. impl Address {
  64. /// Create an address wrapping an IPv4 address with the given octets.
  65. pub fn v4(a0: u8, a1: u8, a2: u8, a3: u8) -> Address {
  66. Address::Ipv4(Ipv4Address::new(a0, a1, a2, a3))
  67. }
  68. /// Query whether the address is a valid unicast address.
  69. pub fn is_unicast(&self) -> bool {
  70. match self {
  71. &Address::Unspecified => false,
  72. &Address::Ipv4(addr) => addr.is_unicast(),
  73. &Address::__Nonexhaustive => unreachable!()
  74. }
  75. }
  76. /// Query whether the address is the broadcast address.
  77. pub fn is_broadcast(&self) -> bool {
  78. match self {
  79. &Address::Unspecified => false,
  80. &Address::Ipv4(addr) => addr.is_broadcast(),
  81. &Address::__Nonexhaustive => unreachable!()
  82. }
  83. }
  84. /// Query whether the address falls into the "unspecified" range.
  85. pub fn is_unspecified(&self) -> bool {
  86. match self {
  87. &Address::Unspecified => true,
  88. &Address::Ipv4(addr) => addr.is_unspecified(),
  89. &Address::__Nonexhaustive => unreachable!()
  90. }
  91. }
  92. /// Return an unspecified address that has the same IP version as `self`.
  93. pub fn to_unspecified(&self) -> Address {
  94. match self {
  95. &Address::Unspecified => Address::Unspecified,
  96. &Address::Ipv4(_) => Address::Ipv4(Ipv4Address::UNSPECIFIED),
  97. &Address::__Nonexhaustive => unreachable!()
  98. }
  99. }
  100. }
  101. impl Default for Address {
  102. fn default() -> Address {
  103. Address::Unspecified
  104. }
  105. }
  106. impl From<Ipv4Address> for Address {
  107. fn from(addr: Ipv4Address) -> Self {
  108. Address::Ipv4(addr)
  109. }
  110. }
  111. impl fmt::Display for Address {
  112. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  113. match self {
  114. &Address::Unspecified => write!(f, "*"),
  115. &Address::Ipv4(addr) => write!(f, "{}", addr),
  116. &Address::__Nonexhaustive => unreachable!()
  117. }
  118. }
  119. }
  120. /// An internet endpoint address.
  121. ///
  122. /// An endpoint can be constructed from a port, in which case the address is unspecified.
  123. #[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Clone, Copy, Default)]
  124. pub struct Endpoint {
  125. pub addr: Address,
  126. pub port: u16
  127. }
  128. impl Endpoint {
  129. /// An endpoint with unspecified address and port.
  130. pub const UNSPECIFIED: Endpoint = Endpoint { addr: Address::Unspecified, port: 0 };
  131. /// Create an endpoint address from given address and port.
  132. pub fn new(addr: Address, port: u16) -> Endpoint {
  133. Endpoint { addr: addr, port: port }
  134. }
  135. /// Query whether the endpoint has a specified address and port.
  136. pub fn is_specified(&self) -> bool {
  137. !self.addr.is_unspecified() && self.port != 0
  138. }
  139. }
  140. impl fmt::Display for Endpoint {
  141. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  142. write!(f, "{}:{}", self.addr, self.port)
  143. }
  144. }
  145. impl From<u16> for Endpoint {
  146. fn from(port: u16) -> Endpoint {
  147. Endpoint { addr: Address::Unspecified, port: port }
  148. }
  149. }
  150. impl<T: Into<Address>> From<(T, u16)> for Endpoint {
  151. fn from((addr, port): (T, u16)) -> Endpoint {
  152. Endpoint { addr: addr.into(), port: port }
  153. }
  154. }
  155. /// An IP packet representation.
  156. ///
  157. /// This enum abstracts the various versions of IP packets. It either contains a concrete
  158. /// high-level representation for some IP protocol version, or an unspecified representation,
  159. /// which permits the `IpAddress::Unspecified` addresses.
  160. #[derive(Debug, Clone, PartialEq, Eq)]
  161. pub enum IpRepr {
  162. Unspecified {
  163. src_addr: Address,
  164. dst_addr: Address,
  165. protocol: Protocol,
  166. payload_len: usize
  167. },
  168. Ipv4(Ipv4Repr),
  169. #[doc(hidden)]
  170. __Nonexhaustive
  171. }
  172. impl From<Ipv4Repr> for IpRepr {
  173. fn from(repr: Ipv4Repr) -> IpRepr {
  174. IpRepr::Ipv4(repr)
  175. }
  176. }
  177. impl IpRepr {
  178. /// Return the protocol version.
  179. pub fn version(&self) -> Version {
  180. match self {
  181. &IpRepr::Unspecified { .. } => Version::Unspecified,
  182. &IpRepr::Ipv4(_) => Version::Ipv4,
  183. &IpRepr::__Nonexhaustive => unreachable!()
  184. }
  185. }
  186. /// Return the source address.
  187. pub fn src_addr(&self) -> Address {
  188. match self {
  189. &IpRepr::Unspecified { src_addr, .. } => src_addr,
  190. &IpRepr::Ipv4(repr) => Address::Ipv4(repr.src_addr),
  191. &IpRepr::__Nonexhaustive => unreachable!()
  192. }
  193. }
  194. /// Return the destination address.
  195. pub fn dst_addr(&self) -> Address {
  196. match self {
  197. &IpRepr::Unspecified { dst_addr, .. } => dst_addr,
  198. &IpRepr::Ipv4(repr) => Address::Ipv4(repr.dst_addr),
  199. &IpRepr::__Nonexhaustive => unreachable!()
  200. }
  201. }
  202. /// Return the protocol.
  203. pub fn protocol(&self) -> Protocol {
  204. match self {
  205. &IpRepr::Unspecified { protocol, .. } => protocol,
  206. &IpRepr::Ipv4(repr) => repr.protocol,
  207. &IpRepr::__Nonexhaustive => unreachable!()
  208. }
  209. }
  210. /// Return the payload length.
  211. pub fn payload_len(&self) -> usize {
  212. match self {
  213. &IpRepr::Unspecified { payload_len, .. } => payload_len,
  214. &IpRepr::Ipv4(repr) => repr.payload_len,
  215. &IpRepr::__Nonexhaustive => unreachable!()
  216. }
  217. }
  218. /// Set the payload length.
  219. pub fn set_payload_len(&mut self, length: usize) {
  220. match self {
  221. &mut IpRepr::Unspecified { ref mut payload_len, .. } =>
  222. *payload_len = length,
  223. &mut IpRepr::Ipv4(Ipv4Repr { ref mut payload_len, .. }) =>
  224. *payload_len = length,
  225. &mut IpRepr::__Nonexhaustive => unreachable!()
  226. }
  227. }
  228. /// Convert an unspecified representation into a concrete one, or return
  229. /// `Err(Error::Unaddressable)` if not possible.
  230. ///
  231. /// # Panics
  232. /// This function panics if source and destination addresses belong to different families,
  233. /// or the destination address is unspecified, since this indicates a logic error.
  234. pub fn lower(&self, fallback_src_addrs: &[Address]) -> Result<IpRepr> {
  235. match self {
  236. &IpRepr::Unspecified {
  237. src_addr: Address::Ipv4(src_addr),
  238. dst_addr: Address::Ipv4(dst_addr),
  239. protocol, payload_len
  240. } => {
  241. Ok(IpRepr::Ipv4(Ipv4Repr {
  242. src_addr: src_addr,
  243. dst_addr: dst_addr,
  244. protocol: protocol,
  245. payload_len: payload_len
  246. }))
  247. }
  248. &IpRepr::Unspecified {
  249. src_addr: Address::Unspecified,
  250. dst_addr: Address::Ipv4(dst_addr),
  251. protocol, payload_len
  252. } => {
  253. let mut src_addr = None;
  254. for addr in fallback_src_addrs {
  255. match addr {
  256. &Address::Ipv4(addr) => {
  257. src_addr = Some(addr);
  258. break
  259. }
  260. _ => ()
  261. }
  262. }
  263. Ok(IpRepr::Ipv4(Ipv4Repr {
  264. src_addr: src_addr.ok_or(Error::Unaddressable)?,
  265. dst_addr: dst_addr,
  266. protocol: protocol,
  267. payload_len: payload_len
  268. }))
  269. }
  270. &IpRepr::Unspecified { dst_addr: Address::Unspecified, .. } =>
  271. panic!("unspecified destination IP address"),
  272. // &IpRepr::Unspecified { .. } =>
  273. // panic!("source and destination IP address families do not match"),
  274. &IpRepr::Ipv4(mut repr) => {
  275. if repr.src_addr.is_unspecified() {
  276. for addr in fallback_src_addrs {
  277. match addr {
  278. &Address::Ipv4(addr) => {
  279. repr.src_addr = addr;
  280. return Ok(IpRepr::Ipv4(repr));
  281. }
  282. _ => ()
  283. }
  284. }
  285. Err(Error::Unaddressable)
  286. } else {
  287. Ok(IpRepr::Ipv4(repr))
  288. }
  289. },
  290. &IpRepr::__Nonexhaustive |
  291. &IpRepr::Unspecified { src_addr: Address::__Nonexhaustive, .. } |
  292. &IpRepr::Unspecified { dst_addr: Address::__Nonexhaustive, .. } =>
  293. unreachable!()
  294. }
  295. }
  296. /// Return the length of a header that will be emitted from this high-level representation.
  297. ///
  298. /// # Panics
  299. /// This function panics if invoked on an unspecified representation.
  300. pub fn buffer_len(&self) -> usize {
  301. match self {
  302. &IpRepr::Unspecified { .. } =>
  303. panic!("unspecified IP representation"),
  304. &IpRepr::Ipv4(repr) =>
  305. repr.buffer_len(),
  306. &IpRepr::__Nonexhaustive =>
  307. unreachable!()
  308. }
  309. }
  310. /// Emit this high-level representation into a buffer.
  311. ///
  312. /// # Panics
  313. /// This function panics if invoked on an unspecified representation.
  314. pub fn emit<T: AsRef<[u8]> + AsMut<[u8]>>(&self, buffer: T, checksum_caps: &ChecksumCapabilities) {
  315. match self {
  316. &IpRepr::Unspecified { .. } =>
  317. panic!("unspecified IP representation"),
  318. &IpRepr::Ipv4(repr) =>
  319. repr.emit(&mut Ipv4Packet::new(buffer), &checksum_caps),
  320. &IpRepr::__Nonexhaustive =>
  321. unreachable!()
  322. }
  323. }
  324. /// Return the total length of a packet that will be emitted from this
  325. /// high-level representation.
  326. ///
  327. /// This is the same as `repr.buffer_len() + repr.payload_len()`.
  328. ///
  329. /// # Panics
  330. /// This function panics if invoked on an unspecified representation.
  331. pub fn total_len(&self) -> usize {
  332. self.buffer_len() + self.payload_len()
  333. }
  334. }
  335. pub mod checksum {
  336. use byteorder::{ByteOrder, NetworkEndian};
  337. use super::*;
  338. fn propagate_carries(word: u32) -> u16 {
  339. let sum = (word >> 16) + (word & 0xffff);
  340. ((sum >> 16) as u16) + (sum as u16)
  341. }
  342. /// Compute an RFC 1071 compliant checksum (without the final complement).
  343. pub fn data(data: &[u8]) -> u16 {
  344. let mut accum: u32 = 0;
  345. let mut i = 0;
  346. while i < data.len() {
  347. let word;
  348. if i + 2 <= data.len() {
  349. word = NetworkEndian::read_u16(&data[i..i + 2]) as u32
  350. } else {
  351. word = (data[i] as u32) << 8
  352. }
  353. accum += word;
  354. i += 2;
  355. }
  356. propagate_carries(accum)
  357. }
  358. /// Combine several RFC 1071 compliant checksums.
  359. pub fn combine(checksums: &[u16]) -> u16 {
  360. let mut accum: u32 = 0;
  361. for &word in checksums {
  362. accum += word as u32;
  363. }
  364. propagate_carries(accum)
  365. }
  366. /// Compute an IP pseudo header checksum.
  367. pub fn pseudo_header(src_addr: &Address, dst_addr: &Address,
  368. protocol: Protocol, length: u32) -> u16 {
  369. match (src_addr, dst_addr) {
  370. (&Address::Ipv4(src_addr), &Address::Ipv4(dst_addr)) => {
  371. let mut proto_len = [0u8; 4];
  372. proto_len[1] = protocol.into();
  373. NetworkEndian::write_u16(&mut proto_len[2..4], length as u16);
  374. combine(&[
  375. data(src_addr.as_bytes()),
  376. data(dst_addr.as_bytes()),
  377. data(&proto_len[..])
  378. ])
  379. },
  380. _ => panic!("Unexpected pseudo header addresses: {}, {}",
  381. src_addr, dst_addr)
  382. }
  383. }
  384. // We use this in pretty printer implementations.
  385. pub(crate) fn write_checksum(f: &mut fmt::Formatter, correct: bool) -> fmt::Result {
  386. if !correct {
  387. write!(f, " (checksum incorrect)")?;
  388. }
  389. write!(f, "\n")
  390. }
  391. }
  392. #[cfg(test)]
  393. mod test {
  394. use super::*;
  395. use wire::{Ipv4Address, IpProtocol, IpAddress, Ipv4Repr};
  396. #[test]
  397. fn ip_repr_lower() {
  398. let ip_addr_a = Ipv4Address::new(1, 2, 3, 4);
  399. let ip_addr_b = Ipv4Address::new(5, 6, 7, 8);
  400. let proto = IpProtocol::Icmp;
  401. let payload_len = 10;
  402. assert_eq!(
  403. IpRepr::Unspecified{
  404. src_addr: IpAddress::Ipv4(ip_addr_a),
  405. dst_addr: IpAddress::Ipv4(ip_addr_b),
  406. protocol: proto,
  407. payload_len
  408. }.lower(&[]),
  409. Ok(IpRepr::Ipv4(Ipv4Repr{
  410. src_addr: ip_addr_a,
  411. dst_addr: ip_addr_b,
  412. protocol: proto,
  413. payload_len
  414. }))
  415. );
  416. assert_eq!(
  417. IpRepr::Unspecified{
  418. src_addr: IpAddress::Unspecified,
  419. dst_addr: IpAddress::Ipv4(ip_addr_b),
  420. protocol: proto,
  421. payload_len
  422. }.lower(&[]),
  423. Err(Error::Unaddressable)
  424. );
  425. assert_eq!(
  426. IpRepr::Unspecified{
  427. src_addr: IpAddress::Unspecified,
  428. dst_addr: IpAddress::Ipv4(ip_addr_b),
  429. protocol: proto,
  430. payload_len
  431. }.lower(&[IpAddress::Ipv4(ip_addr_a)]),
  432. Ok(IpRepr::Ipv4(Ipv4Repr{
  433. src_addr: ip_addr_a,
  434. dst_addr: ip_addr_b,
  435. protocol: proto,
  436. payload_len
  437. }))
  438. );
  439. assert_eq!(
  440. IpRepr::Ipv4(Ipv4Repr{
  441. src_addr: ip_addr_a,
  442. dst_addr: ip_addr_b,
  443. protocol: proto,
  444. payload_len
  445. }).lower(&[]),
  446. Ok(IpRepr::Ipv4(Ipv4Repr{
  447. src_addr: ip_addr_a,
  448. dst_addr: ip_addr_b,
  449. protocol: proto,
  450. payload_len
  451. }))
  452. );
  453. assert_eq!(
  454. IpRepr::Ipv4(Ipv4Repr{
  455. src_addr: Ipv4Address::UNSPECIFIED,
  456. dst_addr: ip_addr_b,
  457. protocol: proto,
  458. payload_len
  459. }).lower(&[]),
  460. Err(Error::Unaddressable)
  461. );
  462. assert_eq!(
  463. IpRepr::Ipv4(Ipv4Repr{
  464. src_addr: Ipv4Address::UNSPECIFIED,
  465. dst_addr: ip_addr_b,
  466. protocol: proto,
  467. payload_len
  468. }).lower(&[IpAddress::Ipv4(ip_addr_a)]),
  469. Ok(IpRepr::Ipv4(Ipv4Repr{
  470. src_addr: ip_addr_a,
  471. dst_addr: ip_addr_b,
  472. protocol: proto,
  473. payload_len
  474. }))
  475. );
  476. }
  477. #[test]
  478. fn endpoint_unspecified() {
  479. assert!(!Endpoint::UNSPECIFIED.is_specified());
  480. }
  481. }