ip.rs 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994
  1. use core::convert::From;
  2. use core::fmt;
  3. use super::{Error, Result};
  4. use crate::phy::ChecksumCapabilities;
  5. #[cfg(feature = "proto-ipv4")]
  6. use crate::wire::{Ipv4Address, Ipv4Cidr, Ipv4Packet, Ipv4Repr};
  7. #[cfg(feature = "proto-ipv6")]
  8. use crate::wire::{Ipv6Address, Ipv6Cidr, Ipv6Packet, Ipv6Repr};
  9. /// Internet protocol version.
  10. #[derive(Debug, Hash, PartialEq, Eq, PartialOrd, Ord, Clone, Copy)]
  11. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  12. pub enum Version {
  13. #[cfg(feature = "proto-ipv4")]
  14. Ipv4,
  15. #[cfg(feature = "proto-ipv6")]
  16. Ipv6,
  17. }
  18. impl Version {
  19. /// Return the version of an IP packet stored in the provided buffer.
  20. ///
  21. /// This function never returns `Ok(IpVersion::Unspecified)`; instead,
  22. /// unknown versions result in `Err(Error)`.
  23. pub const fn of_packet(data: &[u8]) -> Result<Version> {
  24. match data[0] >> 4 {
  25. #[cfg(feature = "proto-ipv4")]
  26. 4 => Ok(Version::Ipv4),
  27. #[cfg(feature = "proto-ipv6")]
  28. 6 => Ok(Version::Ipv6),
  29. _ => Err(Error),
  30. }
  31. }
  32. }
  33. impl fmt::Display for Version {
  34. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  35. match *self {
  36. #[cfg(feature = "proto-ipv4")]
  37. Version::Ipv4 => write!(f, "IPv4"),
  38. #[cfg(feature = "proto-ipv6")]
  39. Version::Ipv6 => write!(f, "IPv6"),
  40. }
  41. }
  42. }
  43. enum_with_unknown! {
  44. /// IP datagram encapsulated protocol.
  45. pub enum Protocol(u8) {
  46. HopByHop = 0x00,
  47. Icmp = 0x01,
  48. Igmp = 0x02,
  49. Tcp = 0x06,
  50. Udp = 0x11,
  51. Ipv6Route = 0x2b,
  52. Ipv6Frag = 0x2c,
  53. IpSecEsp = 0x32,
  54. IpSecAh = 0x33,
  55. Icmpv6 = 0x3a,
  56. Ipv6NoNxt = 0x3b,
  57. Ipv6Opts = 0x3c
  58. }
  59. }
  60. impl fmt::Display for Protocol {
  61. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  62. match *self {
  63. Protocol::HopByHop => write!(f, "Hop-by-Hop"),
  64. Protocol::Icmp => write!(f, "ICMP"),
  65. Protocol::Igmp => write!(f, "IGMP"),
  66. Protocol::Tcp => write!(f, "TCP"),
  67. Protocol::Udp => write!(f, "UDP"),
  68. Protocol::Ipv6Route => write!(f, "IPv6-Route"),
  69. Protocol::Ipv6Frag => write!(f, "IPv6-Frag"),
  70. Protocol::IpSecEsp => write!(f, "IPsec-ESP"),
  71. Protocol::IpSecAh => write!(f, "IPsec-AH"),
  72. Protocol::Icmpv6 => write!(f, "ICMPv6"),
  73. Protocol::Ipv6NoNxt => write!(f, "IPv6-NoNxt"),
  74. Protocol::Ipv6Opts => write!(f, "IPv6-Opts"),
  75. Protocol::Unknown(id) => write!(f, "0x{id:02x}"),
  76. }
  77. }
  78. }
  79. /// An internetworking address.
  80. #[derive(Debug, Hash, PartialEq, Eq, PartialOrd, Ord, Clone, Copy)]
  81. pub enum Address {
  82. /// An IPv4 address.
  83. #[cfg(feature = "proto-ipv4")]
  84. Ipv4(Ipv4Address),
  85. /// An IPv6 address.
  86. #[cfg(feature = "proto-ipv6")]
  87. Ipv6(Ipv6Address),
  88. }
  89. impl Address {
  90. /// Create an address wrapping an IPv4 address with the given octets.
  91. #[cfg(feature = "proto-ipv4")]
  92. pub const fn v4(a0: u8, a1: u8, a2: u8, a3: u8) -> Address {
  93. Address::Ipv4(Ipv4Address::new(a0, a1, a2, a3))
  94. }
  95. /// Create an address wrapping an IPv6 address with the given octets.
  96. #[cfg(feature = "proto-ipv6")]
  97. #[allow(clippy::too_many_arguments)]
  98. pub const fn v6(
  99. a0: u16,
  100. a1: u16,
  101. a2: u16,
  102. a3: u16,
  103. a4: u16,
  104. a5: u16,
  105. a6: u16,
  106. a7: u16,
  107. ) -> Address {
  108. Address::Ipv6(Ipv6Address::new(a0, a1, a2, a3, a4, a5, a6, a7))
  109. }
  110. /// Return the protocol version.
  111. pub const fn version(&self) -> Version {
  112. match self {
  113. #[cfg(feature = "proto-ipv4")]
  114. Address::Ipv4(_) => Version::Ipv4,
  115. #[cfg(feature = "proto-ipv6")]
  116. Address::Ipv6(_) => Version::Ipv6,
  117. }
  118. }
  119. /// Return an address as a sequence of octets, in big-endian.
  120. pub const fn as_bytes(&self) -> &[u8] {
  121. match self {
  122. #[cfg(feature = "proto-ipv4")]
  123. Address::Ipv4(addr) => addr.as_bytes(),
  124. #[cfg(feature = "proto-ipv6")]
  125. Address::Ipv6(addr) => addr.as_bytes(),
  126. }
  127. }
  128. /// Query whether the address is a valid unicast address.
  129. pub fn is_unicast(&self) -> bool {
  130. match self {
  131. #[cfg(feature = "proto-ipv4")]
  132. Address::Ipv4(addr) => addr.is_unicast(),
  133. #[cfg(feature = "proto-ipv6")]
  134. Address::Ipv6(addr) => addr.is_unicast(),
  135. }
  136. }
  137. /// Query whether the address is a valid multicast address.
  138. pub const fn is_multicast(&self) -> bool {
  139. match self {
  140. #[cfg(feature = "proto-ipv4")]
  141. Address::Ipv4(addr) => addr.is_multicast(),
  142. #[cfg(feature = "proto-ipv6")]
  143. Address::Ipv6(addr) => addr.is_multicast(),
  144. }
  145. }
  146. /// Query whether the address is the broadcast address.
  147. pub fn is_broadcast(&self) -> bool {
  148. match self {
  149. #[cfg(feature = "proto-ipv4")]
  150. Address::Ipv4(addr) => addr.is_broadcast(),
  151. #[cfg(feature = "proto-ipv6")]
  152. Address::Ipv6(_) => false,
  153. }
  154. }
  155. /// Query whether the address falls into the "unspecified" range.
  156. pub fn is_unspecified(&self) -> bool {
  157. match self {
  158. #[cfg(feature = "proto-ipv4")]
  159. Address::Ipv4(addr) => addr.is_unspecified(),
  160. #[cfg(feature = "proto-ipv6")]
  161. Address::Ipv6(addr) => addr.is_unspecified(),
  162. }
  163. }
  164. /// If `self` is a CIDR-compatible subnet mask, return `Some(prefix_len)`,
  165. /// where `prefix_len` is the number of leading zeroes. Return `None` otherwise.
  166. pub fn prefix_len(&self) -> Option<u8> {
  167. let mut ones = true;
  168. let mut prefix_len = 0;
  169. for byte in self.as_bytes() {
  170. let mut mask = 0x80;
  171. for _ in 0..8 {
  172. let one = *byte & mask != 0;
  173. if ones {
  174. // Expect 1s until first 0
  175. if one {
  176. prefix_len += 1;
  177. } else {
  178. ones = false;
  179. }
  180. } else if one {
  181. // 1 where 0 was expected
  182. return None;
  183. }
  184. mask >>= 1;
  185. }
  186. }
  187. Some(prefix_len)
  188. }
  189. }
  190. #[cfg(all(feature = "proto-ipv4", feature = "proto-ipv6"))]
  191. impl From<::core::net::IpAddr> for Address {
  192. fn from(x: ::core::net::IpAddr) -> Address {
  193. match x {
  194. ::core::net::IpAddr::V4(ipv4) => Address::Ipv4(ipv4.into()),
  195. ::core::net::IpAddr::V6(ipv6) => Address::Ipv6(ipv6.into()),
  196. }
  197. }
  198. }
  199. impl From<Address> for ::core::net::IpAddr {
  200. fn from(x: Address) -> ::core::net::IpAddr {
  201. match x {
  202. #[cfg(feature = "proto-ipv4")]
  203. Address::Ipv4(ipv4) => ::core::net::IpAddr::V4(ipv4.into()),
  204. #[cfg(feature = "proto-ipv6")]
  205. Address::Ipv6(ipv6) => ::core::net::IpAddr::V6(ipv6.into()),
  206. }
  207. }
  208. }
  209. #[cfg(feature = "proto-ipv4")]
  210. impl From<::core::net::Ipv4Addr> for Address {
  211. fn from(ipv4: ::core::net::Ipv4Addr) -> Address {
  212. Address::Ipv4(ipv4.into())
  213. }
  214. }
  215. #[cfg(feature = "proto-ipv6")]
  216. impl From<::core::net::Ipv6Addr> for Address {
  217. fn from(ipv6: ::core::net::Ipv6Addr) -> Address {
  218. Address::Ipv6(ipv6.into())
  219. }
  220. }
  221. #[cfg(feature = "proto-ipv4")]
  222. impl From<Ipv4Address> for Address {
  223. fn from(addr: Ipv4Address) -> Self {
  224. Address::Ipv4(addr)
  225. }
  226. }
  227. #[cfg(feature = "proto-ipv6")]
  228. impl From<Ipv6Address> for Address {
  229. fn from(addr: Ipv6Address) -> Self {
  230. Address::Ipv6(addr)
  231. }
  232. }
  233. impl fmt::Display for Address {
  234. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  235. match *self {
  236. #[cfg(feature = "proto-ipv4")]
  237. Address::Ipv4(addr) => write!(f, "{addr}"),
  238. #[cfg(feature = "proto-ipv6")]
  239. Address::Ipv6(addr) => write!(f, "{addr}"),
  240. }
  241. }
  242. }
  243. #[cfg(feature = "defmt")]
  244. impl defmt::Format for Address {
  245. fn format(&self, f: defmt::Formatter) {
  246. match self {
  247. #[cfg(feature = "proto-ipv4")]
  248. &Address::Ipv4(addr) => defmt::write!(f, "{:?}", addr),
  249. #[cfg(feature = "proto-ipv6")]
  250. &Address::Ipv6(addr) => defmt::write!(f, "{:?}", addr),
  251. }
  252. }
  253. }
  254. /// A specification of a CIDR block, containing an address and a variable-length
  255. /// subnet masking prefix length.
  256. #[derive(Debug, Hash, PartialEq, Eq, PartialOrd, Ord, Clone, Copy)]
  257. pub enum Cidr {
  258. #[cfg(feature = "proto-ipv4")]
  259. Ipv4(Ipv4Cidr),
  260. #[cfg(feature = "proto-ipv6")]
  261. Ipv6(Ipv6Cidr),
  262. }
  263. impl Cidr {
  264. /// Create a CIDR block from the given address and prefix length.
  265. ///
  266. /// # Panics
  267. /// This function panics if the given prefix length is invalid for the given address.
  268. pub fn new(addr: Address, prefix_len: u8) -> Cidr {
  269. match addr {
  270. #[cfg(feature = "proto-ipv4")]
  271. Address::Ipv4(addr) => Cidr::Ipv4(Ipv4Cidr::new(addr, prefix_len)),
  272. #[cfg(feature = "proto-ipv6")]
  273. Address::Ipv6(addr) => Cidr::Ipv6(Ipv6Cidr::new(addr, prefix_len)),
  274. }
  275. }
  276. /// Return the IP address of this CIDR block.
  277. pub const fn address(&self) -> Address {
  278. match *self {
  279. #[cfg(feature = "proto-ipv4")]
  280. Cidr::Ipv4(cidr) => Address::Ipv4(cidr.address()),
  281. #[cfg(feature = "proto-ipv6")]
  282. Cidr::Ipv6(cidr) => Address::Ipv6(cidr.address()),
  283. }
  284. }
  285. /// Return the prefix length of this CIDR block.
  286. pub const fn prefix_len(&self) -> u8 {
  287. match *self {
  288. #[cfg(feature = "proto-ipv4")]
  289. Cidr::Ipv4(cidr) => cidr.prefix_len(),
  290. #[cfg(feature = "proto-ipv6")]
  291. Cidr::Ipv6(cidr) => cidr.prefix_len(),
  292. }
  293. }
  294. /// Query whether the subnetwork described by this CIDR block contains
  295. /// the given address.
  296. pub fn contains_addr(&self, addr: &Address) -> bool {
  297. match (self, addr) {
  298. #[cfg(feature = "proto-ipv4")]
  299. (Cidr::Ipv4(cidr), Address::Ipv4(addr)) => cidr.contains_addr(addr),
  300. #[cfg(feature = "proto-ipv6")]
  301. (Cidr::Ipv6(cidr), Address::Ipv6(addr)) => cidr.contains_addr(addr),
  302. #[allow(unreachable_patterns)]
  303. _ => false,
  304. }
  305. }
  306. /// Query whether the subnetwork described by this CIDR block contains
  307. /// the subnetwork described by the given CIDR block.
  308. pub fn contains_subnet(&self, subnet: &Cidr) -> bool {
  309. match (self, subnet) {
  310. #[cfg(feature = "proto-ipv4")]
  311. (Cidr::Ipv4(cidr), Cidr::Ipv4(other)) => cidr.contains_subnet(other),
  312. #[cfg(feature = "proto-ipv6")]
  313. (Cidr::Ipv6(cidr), Cidr::Ipv6(other)) => cidr.contains_subnet(other),
  314. #[allow(unreachable_patterns)]
  315. _ => false,
  316. }
  317. }
  318. }
  319. #[cfg(feature = "proto-ipv4")]
  320. impl From<Ipv4Cidr> for Cidr {
  321. fn from(addr: Ipv4Cidr) -> Self {
  322. Cidr::Ipv4(addr)
  323. }
  324. }
  325. #[cfg(feature = "proto-ipv6")]
  326. impl From<Ipv6Cidr> for Cidr {
  327. fn from(addr: Ipv6Cidr) -> Self {
  328. Cidr::Ipv6(addr)
  329. }
  330. }
  331. impl fmt::Display for Cidr {
  332. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  333. match *self {
  334. #[cfg(feature = "proto-ipv4")]
  335. Cidr::Ipv4(cidr) => write!(f, "{cidr}"),
  336. #[cfg(feature = "proto-ipv6")]
  337. Cidr::Ipv6(cidr) => write!(f, "{cidr}"),
  338. }
  339. }
  340. }
  341. #[cfg(feature = "defmt")]
  342. impl defmt::Format for Cidr {
  343. fn format(&self, f: defmt::Formatter) {
  344. match self {
  345. #[cfg(feature = "proto-ipv4")]
  346. &Cidr::Ipv4(cidr) => defmt::write!(f, "{:?}", cidr),
  347. #[cfg(feature = "proto-ipv6")]
  348. &Cidr::Ipv6(cidr) => defmt::write!(f, "{:?}", cidr),
  349. }
  350. }
  351. }
  352. /// An internet endpoint address.
  353. ///
  354. /// `Endpoint` always fully specifies both the address and the port.
  355. ///
  356. /// See also ['ListenEndpoint'], which allows not specifying the address
  357. /// in order to listen on a given port on any address.
  358. #[derive(Debug, Hash, PartialEq, Eq, PartialOrd, Ord, Clone, Copy)]
  359. pub struct Endpoint {
  360. pub addr: Address,
  361. pub port: u16,
  362. }
  363. impl Endpoint {
  364. /// Create an endpoint address from given address and port.
  365. pub const fn new(addr: Address, port: u16) -> Endpoint {
  366. Endpoint { addr, port }
  367. }
  368. }
  369. #[cfg(all(feature = "proto-ipv4", feature = "proto-ipv6"))]
  370. impl From<::core::net::SocketAddr> for Endpoint {
  371. fn from(x: ::core::net::SocketAddr) -> Endpoint {
  372. Endpoint {
  373. addr: x.ip().into(),
  374. port: x.port(),
  375. }
  376. }
  377. }
  378. #[cfg(feature = "proto-ipv4")]
  379. impl From<::core::net::SocketAddrV4> for Endpoint {
  380. fn from(x: ::core::net::SocketAddrV4) -> Endpoint {
  381. Endpoint {
  382. addr: (*x.ip()).into(),
  383. port: x.port(),
  384. }
  385. }
  386. }
  387. #[cfg(feature = "proto-ipv6")]
  388. impl From<::core::net::SocketAddrV6> for Endpoint {
  389. fn from(x: ::core::net::SocketAddrV6) -> Endpoint {
  390. Endpoint {
  391. addr: (*x.ip()).into(),
  392. port: x.port(),
  393. }
  394. }
  395. }
  396. impl fmt::Display for Endpoint {
  397. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  398. write!(f, "{}:{}", self.addr, self.port)
  399. }
  400. }
  401. #[cfg(feature = "defmt")]
  402. impl defmt::Format for Endpoint {
  403. fn format(&self, f: defmt::Formatter) {
  404. defmt::write!(f, "{:?}:{=u16}", self.addr, self.port);
  405. }
  406. }
  407. impl<T: Into<Address>> From<(T, u16)> for Endpoint {
  408. fn from((addr, port): (T, u16)) -> Endpoint {
  409. Endpoint {
  410. addr: addr.into(),
  411. port,
  412. }
  413. }
  414. }
  415. /// An internet endpoint address for listening.
  416. ///
  417. /// In contrast with [`Endpoint`], `ListenEndpoint` allows not specifying the address,
  418. /// in order to listen on a given port at all our addresses.
  419. ///
  420. /// An endpoint can be constructed from a port, in which case the address is unspecified.
  421. #[derive(Debug, Hash, PartialEq, Eq, PartialOrd, Ord, Clone, Copy, Default)]
  422. pub struct ListenEndpoint {
  423. pub addr: Option<Address>,
  424. pub port: u16,
  425. }
  426. impl ListenEndpoint {
  427. /// Query whether the endpoint has a specified address and port.
  428. pub const fn is_specified(&self) -> bool {
  429. self.addr.is_some() && self.port != 0
  430. }
  431. }
  432. #[cfg(all(feature = "proto-ipv4", feature = "proto-ipv6"))]
  433. impl From<::core::net::SocketAddr> for ListenEndpoint {
  434. fn from(x: ::core::net::SocketAddr) -> ListenEndpoint {
  435. ListenEndpoint {
  436. addr: Some(x.ip().into()),
  437. port: x.port(),
  438. }
  439. }
  440. }
  441. #[cfg(feature = "proto-ipv4")]
  442. impl From<::core::net::SocketAddrV4> for ListenEndpoint {
  443. fn from(x: ::core::net::SocketAddrV4) -> ListenEndpoint {
  444. ListenEndpoint {
  445. addr: Some((*x.ip()).into()),
  446. port: x.port(),
  447. }
  448. }
  449. }
  450. #[cfg(feature = "proto-ipv6")]
  451. impl From<::core::net::SocketAddrV6> for ListenEndpoint {
  452. fn from(x: ::core::net::SocketAddrV6) -> ListenEndpoint {
  453. ListenEndpoint {
  454. addr: Some((*x.ip()).into()),
  455. port: x.port(),
  456. }
  457. }
  458. }
  459. impl fmt::Display for ListenEndpoint {
  460. fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
  461. if let Some(addr) = self.addr {
  462. write!(f, "{}:{}", addr, self.port)
  463. } else {
  464. write!(f, "*:{}", self.port)
  465. }
  466. }
  467. }
  468. #[cfg(feature = "defmt")]
  469. impl defmt::Format for ListenEndpoint {
  470. fn format(&self, f: defmt::Formatter) {
  471. defmt::write!(f, "{:?}:{=u16}", self.addr, self.port);
  472. }
  473. }
  474. impl From<u16> for ListenEndpoint {
  475. fn from(port: u16) -> ListenEndpoint {
  476. ListenEndpoint { addr: None, port }
  477. }
  478. }
  479. impl From<Endpoint> for ListenEndpoint {
  480. fn from(endpoint: Endpoint) -> ListenEndpoint {
  481. ListenEndpoint {
  482. addr: Some(endpoint.addr),
  483. port: endpoint.port,
  484. }
  485. }
  486. }
  487. impl<T: Into<Address>> From<(T, u16)> for ListenEndpoint {
  488. fn from((addr, port): (T, u16)) -> ListenEndpoint {
  489. ListenEndpoint {
  490. addr: Some(addr.into()),
  491. port,
  492. }
  493. }
  494. }
  495. /// An IP packet representation.
  496. ///
  497. /// This enum abstracts the various versions of IP packets. It either contains an IPv4
  498. /// or IPv6 concrete high-level representation.
  499. #[derive(Debug, Clone, PartialEq, Eq)]
  500. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  501. pub enum Repr {
  502. #[cfg(feature = "proto-ipv4")]
  503. Ipv4(Ipv4Repr),
  504. #[cfg(feature = "proto-ipv6")]
  505. Ipv6(Ipv6Repr),
  506. }
  507. #[cfg(feature = "proto-ipv4")]
  508. impl From<Ipv4Repr> for Repr {
  509. fn from(repr: Ipv4Repr) -> Repr {
  510. Repr::Ipv4(repr)
  511. }
  512. }
  513. #[cfg(feature = "proto-ipv6")]
  514. impl From<Ipv6Repr> for Repr {
  515. fn from(repr: Ipv6Repr) -> Repr {
  516. Repr::Ipv6(repr)
  517. }
  518. }
  519. impl Repr {
  520. /// Create a new IpRepr, choosing the right IP version for the src/dst addrs.
  521. ///
  522. /// # Panics
  523. ///
  524. /// Panics if `src_addr` and `dst_addr` are different IP version.
  525. pub fn new(
  526. src_addr: Address,
  527. dst_addr: Address,
  528. next_header: Protocol,
  529. payload_len: usize,
  530. hop_limit: u8,
  531. ) -> Self {
  532. match (src_addr, dst_addr) {
  533. #[cfg(feature = "proto-ipv4")]
  534. (Address::Ipv4(src_addr), Address::Ipv4(dst_addr)) => Self::Ipv4(Ipv4Repr {
  535. src_addr,
  536. dst_addr,
  537. next_header,
  538. payload_len,
  539. hop_limit,
  540. }),
  541. #[cfg(feature = "proto-ipv6")]
  542. (Address::Ipv6(src_addr), Address::Ipv6(dst_addr)) => Self::Ipv6(Ipv6Repr {
  543. src_addr,
  544. dst_addr,
  545. next_header,
  546. payload_len,
  547. hop_limit,
  548. }),
  549. #[allow(unreachable_patterns)]
  550. _ => panic!("IP version mismatch: src={src_addr:?} dst={dst_addr:?}"),
  551. }
  552. }
  553. /// Return the protocol version.
  554. pub const fn version(&self) -> Version {
  555. match *self {
  556. #[cfg(feature = "proto-ipv4")]
  557. Repr::Ipv4(_) => Version::Ipv4,
  558. #[cfg(feature = "proto-ipv6")]
  559. Repr::Ipv6(_) => Version::Ipv6,
  560. }
  561. }
  562. /// Return the source address.
  563. pub const fn src_addr(&self) -> Address {
  564. match *self {
  565. #[cfg(feature = "proto-ipv4")]
  566. Repr::Ipv4(repr) => Address::Ipv4(repr.src_addr),
  567. #[cfg(feature = "proto-ipv6")]
  568. Repr::Ipv6(repr) => Address::Ipv6(repr.src_addr),
  569. }
  570. }
  571. /// Return the destination address.
  572. pub const fn dst_addr(&self) -> Address {
  573. match *self {
  574. #[cfg(feature = "proto-ipv4")]
  575. Repr::Ipv4(repr) => Address::Ipv4(repr.dst_addr),
  576. #[cfg(feature = "proto-ipv6")]
  577. Repr::Ipv6(repr) => Address::Ipv6(repr.dst_addr),
  578. }
  579. }
  580. /// Return the next header (protocol).
  581. pub const fn next_header(&self) -> Protocol {
  582. match *self {
  583. #[cfg(feature = "proto-ipv4")]
  584. Repr::Ipv4(repr) => repr.next_header,
  585. #[cfg(feature = "proto-ipv6")]
  586. Repr::Ipv6(repr) => repr.next_header,
  587. }
  588. }
  589. /// Return the payload length.
  590. pub const fn payload_len(&self) -> usize {
  591. match *self {
  592. #[cfg(feature = "proto-ipv4")]
  593. Repr::Ipv4(repr) => repr.payload_len,
  594. #[cfg(feature = "proto-ipv6")]
  595. Repr::Ipv6(repr) => repr.payload_len,
  596. }
  597. }
  598. /// Set the payload length.
  599. pub fn set_payload_len(&mut self, length: usize) {
  600. match self {
  601. #[cfg(feature = "proto-ipv4")]
  602. Repr::Ipv4(Ipv4Repr { payload_len, .. }) => *payload_len = length,
  603. #[cfg(feature = "proto-ipv6")]
  604. Repr::Ipv6(Ipv6Repr { payload_len, .. }) => *payload_len = length,
  605. }
  606. }
  607. /// Return the TTL value.
  608. pub const fn hop_limit(&self) -> u8 {
  609. match *self {
  610. #[cfg(feature = "proto-ipv4")]
  611. Repr::Ipv4(Ipv4Repr { hop_limit, .. }) => hop_limit,
  612. #[cfg(feature = "proto-ipv6")]
  613. Repr::Ipv6(Ipv6Repr { hop_limit, .. }) => hop_limit,
  614. }
  615. }
  616. /// Return the length of a header that will be emitted from this high-level representation.
  617. pub const fn header_len(&self) -> usize {
  618. match *self {
  619. #[cfg(feature = "proto-ipv4")]
  620. Repr::Ipv4(repr) => repr.buffer_len(),
  621. #[cfg(feature = "proto-ipv6")]
  622. Repr::Ipv6(repr) => repr.buffer_len(),
  623. }
  624. }
  625. /// Emit this high-level representation into a buffer.
  626. pub fn emit<T: AsRef<[u8]> + AsMut<[u8]>>(
  627. &self,
  628. buffer: T,
  629. _checksum_caps: &ChecksumCapabilities,
  630. ) {
  631. match *self {
  632. #[cfg(feature = "proto-ipv4")]
  633. Repr::Ipv4(repr) => repr.emit(&mut Ipv4Packet::new_unchecked(buffer), _checksum_caps),
  634. #[cfg(feature = "proto-ipv6")]
  635. Repr::Ipv6(repr) => repr.emit(&mut Ipv6Packet::new_unchecked(buffer)),
  636. }
  637. }
  638. /// Return the total length of a packet that will be emitted from this
  639. /// high-level representation.
  640. ///
  641. /// This is the same as `repr.buffer_len() + repr.payload_len()`.
  642. pub const fn buffer_len(&self) -> usize {
  643. self.header_len() + self.payload_len()
  644. }
  645. }
  646. pub mod checksum {
  647. use byteorder::{ByteOrder, NetworkEndian};
  648. use super::*;
  649. const fn propagate_carries(word: u32) -> u16 {
  650. let sum = (word >> 16) + (word & 0xffff);
  651. ((sum >> 16) as u16) + (sum as u16)
  652. }
  653. /// Compute an RFC 1071 compliant checksum (without the final complement).
  654. pub fn data(mut data: &[u8]) -> u16 {
  655. let mut accum = 0;
  656. // For each 32-byte chunk...
  657. const CHUNK_SIZE: usize = 32;
  658. while data.len() >= CHUNK_SIZE {
  659. let mut d = &data[..CHUNK_SIZE];
  660. // ... take by 2 bytes and sum them.
  661. while d.len() >= 2 {
  662. accum += NetworkEndian::read_u16(d) as u32;
  663. d = &d[2..];
  664. }
  665. data = &data[CHUNK_SIZE..];
  666. }
  667. // Sum the rest that does not fit the last 32-byte chunk,
  668. // taking by 2 bytes.
  669. while data.len() >= 2 {
  670. accum += NetworkEndian::read_u16(data) as u32;
  671. data = &data[2..];
  672. }
  673. // Add the last remaining odd byte, if any.
  674. if let Some(&value) = data.first() {
  675. accum += (value as u32) << 8;
  676. }
  677. propagate_carries(accum)
  678. }
  679. /// Combine several RFC 1071 compliant checksums.
  680. pub fn combine(checksums: &[u16]) -> u16 {
  681. let mut accum: u32 = 0;
  682. for &word in checksums {
  683. accum += word as u32;
  684. }
  685. propagate_carries(accum)
  686. }
  687. #[cfg(feature = "proto-ipv4")]
  688. pub fn pseudo_header_v4(
  689. src_addr: &Ipv4Address,
  690. dst_addr: &Ipv4Address,
  691. next_header: Protocol,
  692. length: u32,
  693. ) -> u16 {
  694. let mut proto_len = [0u8; 4];
  695. proto_len[1] = next_header.into();
  696. NetworkEndian::write_u16(&mut proto_len[2..4], length as u16);
  697. combine(&[
  698. data(src_addr.as_bytes()),
  699. data(dst_addr.as_bytes()),
  700. data(&proto_len[..]),
  701. ])
  702. }
  703. #[cfg(feature = "proto-ipv6")]
  704. pub fn pseudo_header_v6(
  705. src_addr: &Ipv6Address,
  706. dst_addr: &Ipv6Address,
  707. next_header: Protocol,
  708. length: u32,
  709. ) -> u16 {
  710. let mut proto_len = [0u8; 4];
  711. proto_len[1] = next_header.into();
  712. NetworkEndian::write_u16(&mut proto_len[2..4], length as u16);
  713. combine(&[
  714. data(src_addr.as_bytes()),
  715. data(dst_addr.as_bytes()),
  716. data(&proto_len[..]),
  717. ])
  718. }
  719. pub fn pseudo_header(
  720. src_addr: &Address,
  721. dst_addr: &Address,
  722. next_header: Protocol,
  723. length: u32,
  724. ) -> u16 {
  725. match (src_addr, dst_addr) {
  726. #[cfg(feature = "proto-ipv4")]
  727. (Address::Ipv4(src_addr), Address::Ipv4(dst_addr)) => {
  728. pseudo_header_v4(src_addr, dst_addr, next_header, length)
  729. }
  730. #[cfg(feature = "proto-ipv6")]
  731. (Address::Ipv6(src_addr), Address::Ipv6(dst_addr)) => {
  732. pseudo_header_v6(src_addr, dst_addr, next_header, length)
  733. }
  734. #[allow(unreachable_patterns)]
  735. _ => unreachable!(),
  736. }
  737. }
  738. // We use this in pretty printer implementations.
  739. pub(crate) fn format_checksum(f: &mut fmt::Formatter, correct: bool) -> fmt::Result {
  740. if !correct {
  741. write!(f, " (checksum incorrect)")
  742. } else {
  743. Ok(())
  744. }
  745. }
  746. }
  747. use crate::wire::pretty_print::PrettyIndent;
  748. pub fn pretty_print_ip_payload<T: Into<Repr>>(
  749. f: &mut fmt::Formatter,
  750. indent: &mut PrettyIndent,
  751. ip_repr: T,
  752. payload: &[u8],
  753. ) -> fmt::Result {
  754. #[cfg(feature = "proto-ipv4")]
  755. use super::pretty_print::PrettyPrint;
  756. use crate::wire::ip::checksum::format_checksum;
  757. #[cfg(feature = "proto-ipv4")]
  758. use crate::wire::Icmpv4Packet;
  759. use crate::wire::{TcpPacket, TcpRepr, UdpPacket, UdpRepr};
  760. let checksum_caps = ChecksumCapabilities::ignored();
  761. let repr = ip_repr.into();
  762. match repr.next_header() {
  763. #[cfg(feature = "proto-ipv4")]
  764. Protocol::Icmp => {
  765. indent.increase(f)?;
  766. Icmpv4Packet::<&[u8]>::pretty_print(&payload, f, indent)
  767. }
  768. Protocol::Udp => {
  769. indent.increase(f)?;
  770. match UdpPacket::<&[u8]>::new_checked(payload) {
  771. Err(err) => write!(f, "{indent}({err})"),
  772. Ok(udp_packet) => {
  773. match UdpRepr::parse(
  774. &udp_packet,
  775. &repr.src_addr(),
  776. &repr.dst_addr(),
  777. &checksum_caps,
  778. ) {
  779. Err(err) => write!(f, "{indent}{udp_packet} ({err})"),
  780. Ok(udp_repr) => {
  781. write!(
  782. f,
  783. "{}{} len={}",
  784. indent,
  785. udp_repr,
  786. udp_packet.payload().len()
  787. )?;
  788. let valid =
  789. udp_packet.verify_checksum(&repr.src_addr(), &repr.dst_addr());
  790. format_checksum(f, valid)
  791. }
  792. }
  793. }
  794. }
  795. }
  796. Protocol::Tcp => {
  797. indent.increase(f)?;
  798. match TcpPacket::<&[u8]>::new_checked(payload) {
  799. Err(err) => write!(f, "{indent}({err})"),
  800. Ok(tcp_packet) => {
  801. match TcpRepr::parse(
  802. &tcp_packet,
  803. &repr.src_addr(),
  804. &repr.dst_addr(),
  805. &checksum_caps,
  806. ) {
  807. Err(err) => write!(f, "{indent}{tcp_packet} ({err})"),
  808. Ok(tcp_repr) => {
  809. write!(f, "{indent}{tcp_repr}")?;
  810. let valid =
  811. tcp_packet.verify_checksum(&repr.src_addr(), &repr.dst_addr());
  812. format_checksum(f, valid)
  813. }
  814. }
  815. }
  816. }
  817. }
  818. _ => Ok(()),
  819. }
  820. }
  821. #[cfg(test)]
  822. pub(crate) mod test {
  823. #![allow(unused)]
  824. use super::*;
  825. use crate::wire::{IpAddress, IpCidr, IpProtocol};
  826. #[cfg(feature = "proto-ipv4")]
  827. use crate::wire::{Ipv4Address, Ipv4Repr};
  828. #[test]
  829. #[cfg(feature = "proto-ipv4")]
  830. fn to_prefix_len_ipv4() {
  831. fn test_eq<A: Into<Address>>(prefix_len: u8, mask: A) {
  832. assert_eq!(Some(prefix_len), mask.into().prefix_len());
  833. }
  834. test_eq(0, Ipv4Address::new(0, 0, 0, 0));
  835. test_eq(1, Ipv4Address::new(128, 0, 0, 0));
  836. test_eq(2, Ipv4Address::new(192, 0, 0, 0));
  837. test_eq(3, Ipv4Address::new(224, 0, 0, 0));
  838. test_eq(4, Ipv4Address::new(240, 0, 0, 0));
  839. test_eq(5, Ipv4Address::new(248, 0, 0, 0));
  840. test_eq(6, Ipv4Address::new(252, 0, 0, 0));
  841. test_eq(7, Ipv4Address::new(254, 0, 0, 0));
  842. test_eq(8, Ipv4Address::new(255, 0, 0, 0));
  843. test_eq(9, Ipv4Address::new(255, 128, 0, 0));
  844. test_eq(10, Ipv4Address::new(255, 192, 0, 0));
  845. test_eq(11, Ipv4Address::new(255, 224, 0, 0));
  846. test_eq(12, Ipv4Address::new(255, 240, 0, 0));
  847. test_eq(13, Ipv4Address::new(255, 248, 0, 0));
  848. test_eq(14, Ipv4Address::new(255, 252, 0, 0));
  849. test_eq(15, Ipv4Address::new(255, 254, 0, 0));
  850. test_eq(16, Ipv4Address::new(255, 255, 0, 0));
  851. test_eq(17, Ipv4Address::new(255, 255, 128, 0));
  852. test_eq(18, Ipv4Address::new(255, 255, 192, 0));
  853. test_eq(19, Ipv4Address::new(255, 255, 224, 0));
  854. test_eq(20, Ipv4Address::new(255, 255, 240, 0));
  855. test_eq(21, Ipv4Address::new(255, 255, 248, 0));
  856. test_eq(22, Ipv4Address::new(255, 255, 252, 0));
  857. test_eq(23, Ipv4Address::new(255, 255, 254, 0));
  858. test_eq(24, Ipv4Address::new(255, 255, 255, 0));
  859. test_eq(25, Ipv4Address::new(255, 255, 255, 128));
  860. test_eq(26, Ipv4Address::new(255, 255, 255, 192));
  861. test_eq(27, Ipv4Address::new(255, 255, 255, 224));
  862. test_eq(28, Ipv4Address::new(255, 255, 255, 240));
  863. test_eq(29, Ipv4Address::new(255, 255, 255, 248));
  864. test_eq(30, Ipv4Address::new(255, 255, 255, 252));
  865. test_eq(31, Ipv4Address::new(255, 255, 255, 254));
  866. test_eq(32, Ipv4Address::new(255, 255, 255, 255));
  867. }
  868. #[test]
  869. #[cfg(feature = "proto-ipv4")]
  870. fn to_prefix_len_ipv4_error() {
  871. assert_eq!(
  872. None,
  873. IpAddress::from(Ipv4Address::new(255, 255, 255, 1)).prefix_len()
  874. );
  875. }
  876. #[test]
  877. #[cfg(feature = "proto-ipv6")]
  878. fn to_prefix_len_ipv6() {
  879. fn test_eq<A: Into<Address>>(prefix_len: u8, mask: A) {
  880. assert_eq!(Some(prefix_len), mask.into().prefix_len());
  881. }
  882. test_eq(0, Ipv6Address::new(0, 0, 0, 0, 0, 0, 0, 0));
  883. test_eq(
  884. 128,
  885. Ipv6Address::new(
  886. 0xffff, 0xffff, 0xffff, 0xffff, 0xffff, 0xffff, 0xffff, 0xffff,
  887. ),
  888. );
  889. }
  890. #[test]
  891. #[cfg(feature = "proto-ipv6")]
  892. fn to_prefix_len_ipv6_error() {
  893. assert_eq!(
  894. None,
  895. IpAddress::from(Ipv6Address::new(
  896. 0xffff, 0xffff, 0xffff, 0xffff, 0xffff, 0xffff, 0, 1
  897. ))
  898. .prefix_len()
  899. );
  900. }
  901. }