dhcpv4.rs 51 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266
  1. // See https://tools.ietf.org/html/rfc2131 for the DHCP specification.
  2. use bitflags::bitflags;
  3. use byteorder::{ByteOrder, NetworkEndian};
  4. use crate::wire::arp::Hardware;
  5. use crate::wire::{EthernetAddress, Ipv4Address};
  6. use crate::{Error, Result};
  7. pub const SERVER_PORT: u16 = 67;
  8. pub const CLIENT_PORT: u16 = 68;
  9. pub const MAX_DNS_SERVER_COUNT: usize = 3;
  10. const DHCP_MAGIC_NUMBER: u32 = 0x63825363;
  11. enum_with_unknown! {
  12. /// The possible opcodes of a DHCP packet.
  13. pub enum OpCode(u8) {
  14. Request = 1,
  15. Reply = 2,
  16. }
  17. }
  18. enum_with_unknown! {
  19. /// The possible message types of a DHCP packet.
  20. pub enum MessageType(u8) {
  21. Discover = 1,
  22. Offer = 2,
  23. Request = 3,
  24. Decline = 4,
  25. Ack = 5,
  26. Nak = 6,
  27. Release = 7,
  28. Inform = 8,
  29. }
  30. }
  31. bitflags! {
  32. pub struct Flags: u16 {
  33. const BROADCAST = 0b1000_0000_0000_0000;
  34. }
  35. }
  36. impl MessageType {
  37. fn opcode(&self) -> OpCode {
  38. match *self {
  39. MessageType::Discover
  40. | MessageType::Inform
  41. | MessageType::Request
  42. | MessageType::Decline
  43. | MessageType::Release => OpCode::Request,
  44. MessageType::Offer | MessageType::Ack | MessageType::Nak => OpCode::Reply,
  45. MessageType::Unknown(_) => OpCode::Unknown(0),
  46. }
  47. }
  48. }
  49. /// A representation of a single DHCP option.
  50. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  51. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  52. pub enum DhcpOption<'a> {
  53. EndOfList,
  54. Pad,
  55. MessageType(MessageType),
  56. RequestedIp(Ipv4Address),
  57. ClientIdentifier(EthernetAddress),
  58. ServerIdentifier(Ipv4Address),
  59. IpLeaseTime(u32),
  60. Router(Ipv4Address),
  61. SubnetMask(Ipv4Address),
  62. MaximumDhcpMessageSize(u16),
  63. Other { kind: u8, data: &'a [u8] },
  64. }
  65. impl<'a> DhcpOption<'a> {
  66. pub fn parse(buffer: &'a [u8]) -> Result<(&'a [u8], DhcpOption<'a>)> {
  67. // See https://tools.ietf.org/html/rfc2132 for all possible DHCP options.
  68. let (skip_len, option);
  69. match *buffer.get(0).ok_or(Error::Truncated)? {
  70. field::OPT_END => {
  71. skip_len = 1;
  72. option = DhcpOption::EndOfList;
  73. }
  74. field::OPT_PAD => {
  75. skip_len = 1;
  76. option = DhcpOption::Pad;
  77. }
  78. kind => {
  79. let length = *buffer.get(1).ok_or(Error::Truncated)? as usize;
  80. skip_len = length + 2;
  81. let data = buffer.get(2..skip_len).ok_or(Error::Truncated)?;
  82. match (kind, length) {
  83. (field::OPT_END, _) | (field::OPT_PAD, _) => unreachable!(),
  84. (field::OPT_DHCP_MESSAGE_TYPE, 1) => {
  85. option = DhcpOption::MessageType(MessageType::from(data[0]));
  86. }
  87. (field::OPT_REQUESTED_IP, 4) => {
  88. option = DhcpOption::RequestedIp(Ipv4Address::from_bytes(data));
  89. }
  90. (field::OPT_CLIENT_ID, 7) => {
  91. let hardware_type = Hardware::from(u16::from(data[0]));
  92. if hardware_type != Hardware::Ethernet {
  93. return Err(Error::Unrecognized);
  94. }
  95. option =
  96. DhcpOption::ClientIdentifier(EthernetAddress::from_bytes(&data[1..]));
  97. }
  98. (field::OPT_SERVER_IDENTIFIER, 4) => {
  99. option = DhcpOption::ServerIdentifier(Ipv4Address::from_bytes(data));
  100. }
  101. (field::OPT_ROUTER, 4) => {
  102. option = DhcpOption::Router(Ipv4Address::from_bytes(data));
  103. }
  104. (field::OPT_SUBNET_MASK, 4) => {
  105. option = DhcpOption::SubnetMask(Ipv4Address::from_bytes(data));
  106. }
  107. (field::OPT_MAX_DHCP_MESSAGE_SIZE, 2) => {
  108. option = DhcpOption::MaximumDhcpMessageSize(u16::from_be_bytes([
  109. data[0], data[1],
  110. ]));
  111. }
  112. (field::OPT_IP_LEASE_TIME, 4) => {
  113. option = DhcpOption::IpLeaseTime(u32::from_be_bytes([
  114. data[0], data[1], data[2], data[3],
  115. ]))
  116. }
  117. (_, _) => {
  118. option = DhcpOption::Other {
  119. kind: kind,
  120. data: data,
  121. };
  122. }
  123. }
  124. }
  125. }
  126. Ok((&buffer[skip_len..], option))
  127. }
  128. pub fn buffer_len(&self) -> usize {
  129. match self {
  130. &DhcpOption::EndOfList => 1,
  131. &DhcpOption::Pad => 1,
  132. &DhcpOption::MessageType(_) => 3,
  133. &DhcpOption::ClientIdentifier(eth_addr) => 3 + eth_addr.as_bytes().len(),
  134. &DhcpOption::RequestedIp(ip)
  135. | &DhcpOption::ServerIdentifier(ip)
  136. | &DhcpOption::Router(ip)
  137. | &DhcpOption::SubnetMask(ip) => 2 + ip.as_bytes().len(),
  138. &DhcpOption::MaximumDhcpMessageSize(_) => 4,
  139. &DhcpOption::IpLeaseTime(_) => 6,
  140. &DhcpOption::Other { data, .. } => 2 + data.len(),
  141. }
  142. }
  143. pub fn emit<'b>(&self, buffer: &'b mut [u8]) -> &'b mut [u8] {
  144. let skip_length;
  145. match *self {
  146. DhcpOption::EndOfList => {
  147. skip_length = 1;
  148. buffer[0] = field::OPT_END;
  149. }
  150. DhcpOption::Pad => {
  151. skip_length = 1;
  152. buffer[0] = field::OPT_PAD;
  153. }
  154. _ => {
  155. skip_length = self.buffer_len();
  156. buffer[1] = (skip_length - 2) as u8;
  157. match *self {
  158. DhcpOption::EndOfList | DhcpOption::Pad => unreachable!(),
  159. DhcpOption::MessageType(value) => {
  160. buffer[0] = field::OPT_DHCP_MESSAGE_TYPE;
  161. buffer[2] = value.into();
  162. }
  163. DhcpOption::ClientIdentifier(eth_addr) => {
  164. buffer[0] = field::OPT_CLIENT_ID;
  165. buffer[2] = u16::from(Hardware::Ethernet) as u8;
  166. buffer[3..9].copy_from_slice(eth_addr.as_bytes());
  167. }
  168. DhcpOption::RequestedIp(ip) => {
  169. buffer[0] = field::OPT_REQUESTED_IP;
  170. buffer[2..6].copy_from_slice(ip.as_bytes());
  171. }
  172. DhcpOption::ServerIdentifier(ip) => {
  173. buffer[0] = field::OPT_SERVER_IDENTIFIER;
  174. buffer[2..6].copy_from_slice(ip.as_bytes());
  175. }
  176. DhcpOption::Router(ip) => {
  177. buffer[0] = field::OPT_ROUTER;
  178. buffer[2..6].copy_from_slice(ip.as_bytes());
  179. }
  180. DhcpOption::SubnetMask(mask) => {
  181. buffer[0] = field::OPT_SUBNET_MASK;
  182. buffer[2..6].copy_from_slice(mask.as_bytes());
  183. }
  184. DhcpOption::MaximumDhcpMessageSize(size) => {
  185. buffer[0] = field::OPT_MAX_DHCP_MESSAGE_SIZE;
  186. buffer[2..4].copy_from_slice(&size.to_be_bytes()[..]);
  187. }
  188. DhcpOption::IpLeaseTime(lease_time) => {
  189. buffer[0] = field::OPT_IP_LEASE_TIME;
  190. buffer[2..6].copy_from_slice(&lease_time.to_be_bytes()[..]);
  191. }
  192. DhcpOption::Other {
  193. kind,
  194. data: provided,
  195. } => {
  196. buffer[0] = kind;
  197. buffer[2..skip_length].copy_from_slice(provided);
  198. }
  199. }
  200. }
  201. }
  202. &mut buffer[skip_length..]
  203. }
  204. }
  205. /// A read/write wrapper around a Dynamic Host Configuration Protocol packet buffer.
  206. #[derive(Debug, PartialEq)]
  207. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  208. pub struct Packet<T: AsRef<[u8]>> {
  209. buffer: T,
  210. }
  211. pub(crate) mod field {
  212. #![allow(non_snake_case)]
  213. #![allow(unused)]
  214. use crate::wire::field::*;
  215. pub const OP: usize = 0;
  216. pub const HTYPE: usize = 1;
  217. pub const HLEN: usize = 2;
  218. pub const HOPS: usize = 3;
  219. pub const XID: Field = 4..8;
  220. pub const SECS: Field = 8..10;
  221. pub const FLAGS: Field = 10..12;
  222. pub const CIADDR: Field = 12..16;
  223. pub const YIADDR: Field = 16..20;
  224. pub const SIADDR: Field = 20..24;
  225. pub const GIADDR: Field = 24..28;
  226. pub const CHADDR: Field = 28..34;
  227. pub const SNAME: Field = 34..108;
  228. pub const FILE: Field = 108..236;
  229. pub const MAGIC_NUMBER: Field = 236..240;
  230. pub const OPTIONS: Rest = 240..;
  231. // Vendor Extensions
  232. pub const OPT_END: u8 = 255;
  233. pub const OPT_PAD: u8 = 0;
  234. pub const OPT_SUBNET_MASK: u8 = 1;
  235. pub const OPT_TIME_OFFSET: u8 = 2;
  236. pub const OPT_ROUTER: u8 = 3;
  237. pub const OPT_TIME_SERVER: u8 = 4;
  238. pub const OPT_NAME_SERVER: u8 = 5;
  239. pub const OPT_DOMAIN_NAME_SERVER: u8 = 6;
  240. pub const OPT_LOG_SERVER: u8 = 7;
  241. pub const OPT_COOKIE_SERVER: u8 = 8;
  242. pub const OPT_LPR_SERVER: u8 = 9;
  243. pub const OPT_IMPRESS_SERVER: u8 = 10;
  244. pub const OPT_RESOURCE_LOCATION_SERVER: u8 = 11;
  245. pub const OPT_HOST_NAME: u8 = 12;
  246. pub const OPT_BOOT_FILE_SIZE: u8 = 13;
  247. pub const OPT_MERIT_DUMP: u8 = 14;
  248. pub const OPT_DOMAIN_NAME: u8 = 15;
  249. pub const OPT_SWAP_SERVER: u8 = 16;
  250. pub const OPT_ROOT_PATH: u8 = 17;
  251. pub const OPT_EXTENSIONS_PATH: u8 = 18;
  252. // IP Layer Parameters per Host
  253. pub const OPT_IP_FORWARDING: u8 = 19;
  254. pub const OPT_NON_LOCAL_SOURCE_ROUTING: u8 = 20;
  255. pub const OPT_POLICY_FILTER: u8 = 21;
  256. pub const OPT_MAX_DATAGRAM_REASSEMBLY_SIZE: u8 = 22;
  257. pub const OPT_DEFAULT_TTL: u8 = 23;
  258. pub const OPT_PATH_MTU_AGING_TIMEOUT: u8 = 24;
  259. pub const OPT_PATH_MTU_PLATEU_TABLE: u8 = 25;
  260. // IP Layer Parameters per Interface
  261. pub const OPT_INTERFACE_MTU: u8 = 26;
  262. pub const OPT_ALL_SUBNETS_ARE_LOCAL: u8 = 27;
  263. pub const OPT_BROADCAST_ADDRESS: u8 = 28;
  264. pub const OPT_PERFORM_MASK_DISCOVERY: u8 = 29;
  265. pub const OPT_MASK_SUPPLIER: u8 = 30;
  266. pub const OPT_PERFORM_ROUTER_DISCOVERY: u8 = 31;
  267. pub const OPT_ROUTER_SOLICITATION_ADDRESS: u8 = 32;
  268. pub const OPT_STATIC_ROUTE: u8 = 33;
  269. // Link Layer Parameters per Interface
  270. pub const OPT_TRAILER_ENCAPSULATION: u8 = 34;
  271. pub const OPT_ARP_CACHE_TIMEOUT: u8 = 35;
  272. pub const OPT_ETHERNET_ENCAPSULATION: u8 = 36;
  273. // TCP Parameters
  274. pub const OPT_TCP_DEFAULT_TTL: u8 = 37;
  275. pub const OPT_TCP_KEEPALIVE_INTERVAL: u8 = 38;
  276. pub const OPT_TCP_KEEPALIVE_GARBAGE: u8 = 39;
  277. // Application and Service Parameters
  278. pub const OPT_NIS_DOMAIN: u8 = 40;
  279. pub const OPT_NIS_SERVERS: u8 = 41;
  280. pub const OPT_NTP_SERVERS: u8 = 42;
  281. pub const OPT_VENDOR_SPECIFIC_INFO: u8 = 43;
  282. pub const OPT_NETBIOS_NAME_SERVER: u8 = 44;
  283. pub const OPT_NETBIOS_DISTRIBUTION_SERVER: u8 = 45;
  284. pub const OPT_NETBIOS_NODE_TYPE: u8 = 46;
  285. pub const OPT_NETBIOS_SCOPE: u8 = 47;
  286. pub const OPT_X_WINDOW_FONT_SERVER: u8 = 48;
  287. pub const OPT_X_WINDOW_DISPLAY_MANAGER: u8 = 49;
  288. pub const OPT_NIS_PLUS_DOMAIN: u8 = 64;
  289. pub const OPT_NIS_PLUS_SERVERS: u8 = 65;
  290. pub const OPT_MOBILE_IP_HOME_AGENT: u8 = 68;
  291. pub const OPT_SMTP_SERVER: u8 = 69;
  292. pub const OPT_POP3_SERVER: u8 = 70;
  293. pub const OPT_NNTP_SERVER: u8 = 71;
  294. pub const OPT_WWW_SERVER: u8 = 72;
  295. pub const OPT_FINGER_SERVER: u8 = 73;
  296. pub const OPT_IRC_SERVER: u8 = 74;
  297. pub const OPT_STREETTALK_SERVER: u8 = 75;
  298. pub const OPT_STDA_SERVER: u8 = 76;
  299. // DHCP Extensions
  300. pub const OPT_REQUESTED_IP: u8 = 50;
  301. pub const OPT_IP_LEASE_TIME: u8 = 51;
  302. pub const OPT_OPTION_OVERLOAD: u8 = 52;
  303. pub const OPT_TFTP_SERVER_NAME: u8 = 66;
  304. pub const OPT_BOOTFILE_NAME: u8 = 67;
  305. pub const OPT_DHCP_MESSAGE_TYPE: u8 = 53;
  306. pub const OPT_SERVER_IDENTIFIER: u8 = 54;
  307. pub const OPT_PARAMETER_REQUEST_LIST: u8 = 55;
  308. pub const OPT_MESSAGE: u8 = 56;
  309. pub const OPT_MAX_DHCP_MESSAGE_SIZE: u8 = 57;
  310. pub const OPT_RENEWAL_TIME_VALUE: u8 = 58;
  311. pub const OPT_REBINDING_TIME_VALUE: u8 = 59;
  312. pub const OPT_VENDOR_CLASS_ID: u8 = 60;
  313. pub const OPT_CLIENT_ID: u8 = 61;
  314. }
  315. impl<T: AsRef<[u8]>> Packet<T> {
  316. /// Imbue a raw octet buffer with DHCP packet structure.
  317. pub fn new_unchecked(buffer: T) -> Packet<T> {
  318. Packet { buffer }
  319. }
  320. /// Shorthand for a combination of [new_unchecked] and [check_len].
  321. ///
  322. /// [new_unchecked]: #method.new_unchecked
  323. /// [check_len]: #method.check_len
  324. pub fn new_checked(buffer: T) -> Result<Packet<T>> {
  325. let packet = Self::new_unchecked(buffer);
  326. packet.check_len()?;
  327. Ok(packet)
  328. }
  329. /// Ensure that no accessor method will panic if called.
  330. /// Returns `Err(Error::Truncated)` if the buffer is too short.
  331. ///
  332. /// [set_header_len]: #method.set_header_len
  333. pub fn check_len(&self) -> Result<()> {
  334. let len = self.buffer.as_ref().len();
  335. if len < field::MAGIC_NUMBER.end {
  336. Err(Error::Truncated)
  337. } else {
  338. Ok(())
  339. }
  340. }
  341. /// Consume the packet, returning the underlying buffer.
  342. pub fn into_inner(self) -> T {
  343. self.buffer
  344. }
  345. /// Returns the operation code of this packet.
  346. pub fn opcode(&self) -> OpCode {
  347. let data = self.buffer.as_ref();
  348. OpCode::from(data[field::OP])
  349. }
  350. /// Returns the hardware protocol type (e.g. ethernet).
  351. pub fn hardware_type(&self) -> Hardware {
  352. let data = self.buffer.as_ref();
  353. Hardware::from(u16::from(data[field::HTYPE]))
  354. }
  355. /// Returns the length of a hardware address in bytes (e.g. 6 for ethernet).
  356. pub fn hardware_len(&self) -> u8 {
  357. self.buffer.as_ref()[field::HLEN]
  358. }
  359. /// Returns the transaction ID.
  360. ///
  361. /// The transaction ID (called `xid` in the specification) is a random number used to
  362. /// associate messages and responses between client and server. The number is chosen by
  363. /// the client.
  364. pub fn transaction_id(&self) -> u32 {
  365. let field = &self.buffer.as_ref()[field::XID];
  366. NetworkEndian::read_u32(field)
  367. }
  368. /// Returns the hardware address of the client (called `chaddr` in the specification).
  369. ///
  370. /// Only ethernet is supported by `smoltcp`, so this functions returns
  371. /// an `EthernetAddress`.
  372. pub fn client_hardware_address(&self) -> EthernetAddress {
  373. let field = &self.buffer.as_ref()[field::CHADDR];
  374. EthernetAddress::from_bytes(field)
  375. }
  376. /// Returns the value of the `hops` field.
  377. ///
  378. /// The `hops` field is set to zero by clients and optionally used by relay agents.
  379. pub fn hops(&self) -> u8 {
  380. self.buffer.as_ref()[field::HOPS]
  381. }
  382. /// Returns the value of the `secs` field.
  383. ///
  384. /// The secs field is filled by clients and describes the number of seconds elapsed
  385. /// since client began process.
  386. pub fn secs(&self) -> u16 {
  387. let field = &self.buffer.as_ref()[field::SECS];
  388. NetworkEndian::read_u16(field)
  389. }
  390. /// Returns the value of the `magic cookie` field in the DHCP options.
  391. ///
  392. /// This field should be always be `0x63825363`.
  393. pub fn magic_number(&self) -> u32 {
  394. let field = &self.buffer.as_ref()[field::MAGIC_NUMBER];
  395. NetworkEndian::read_u32(field)
  396. }
  397. /// Returns the Ipv4 address of the client, zero if not set.
  398. ///
  399. /// This corresponds to the `ciaddr` field in the DHCP specification. According to it,
  400. /// this field is “only filled in if client is in `BOUND`, `RENEW` or `REBINDING` state
  401. /// and can respond to ARP requests”.
  402. pub fn client_ip(&self) -> Ipv4Address {
  403. let field = &self.buffer.as_ref()[field::CIADDR];
  404. Ipv4Address::from_bytes(field)
  405. }
  406. /// Returns the value of the `yiaddr` field, zero if not set.
  407. pub fn your_ip(&self) -> Ipv4Address {
  408. let field = &self.buffer.as_ref()[field::YIADDR];
  409. Ipv4Address::from_bytes(field)
  410. }
  411. /// Returns the value of the `siaddr` field, zero if not set.
  412. pub fn server_ip(&self) -> Ipv4Address {
  413. let field = &self.buffer.as_ref()[field::SIADDR];
  414. Ipv4Address::from_bytes(field)
  415. }
  416. /// Returns the value of the `giaddr` field, zero if not set.
  417. pub fn relay_agent_ip(&self) -> Ipv4Address {
  418. let field = &self.buffer.as_ref()[field::GIADDR];
  419. Ipv4Address::from_bytes(field)
  420. }
  421. pub fn flags(&self) -> Flags {
  422. let field = &self.buffer.as_ref()[field::FLAGS];
  423. Flags::from_bits_truncate(NetworkEndian::read_u16(field))
  424. }
  425. }
  426. impl<'a, T: AsRef<[u8]> + ?Sized> Packet<&'a T> {
  427. /// Return a pointer to the options.
  428. #[inline]
  429. pub fn options(&self) -> Result<&'a [u8]> {
  430. let data = self.buffer.as_ref();
  431. data.get(field::OPTIONS).ok_or(Error::Malformed)
  432. }
  433. }
  434. impl<T: AsRef<[u8]> + AsMut<[u8]>> Packet<T> {
  435. /// Sets the optional `sname` (“server name”) and `file` (“boot file name”) fields to zero.
  436. ///
  437. /// The fields are not commonly used, so we set their value always to zero. **This method
  438. /// must be called when creating a packet, otherwise the emitted values for these fields
  439. /// are undefined!**
  440. pub fn set_sname_and_boot_file_to_zero(&mut self) {
  441. let data = self.buffer.as_mut();
  442. for byte in &mut data[field::SNAME] {
  443. *byte = 0;
  444. }
  445. for byte in &mut data[field::FILE] {
  446. *byte = 0;
  447. }
  448. }
  449. /// Sets the `OpCode` for the packet.
  450. pub fn set_opcode(&mut self, value: OpCode) {
  451. let data = self.buffer.as_mut();
  452. data[field::OP] = value.into();
  453. }
  454. /// Sets the hardware address type (only ethernet is supported).
  455. pub fn set_hardware_type(&mut self, value: Hardware) {
  456. let data = self.buffer.as_mut();
  457. let number: u16 = value.into();
  458. assert!(number <= u16::from(u8::max_value())); // TODO: Replace with TryFrom when it's stable
  459. data[field::HTYPE] = number as u8;
  460. }
  461. /// Sets the hardware address length.
  462. ///
  463. /// Only ethernet is supported, so this field should be set to the value `6`.
  464. pub fn set_hardware_len(&mut self, value: u8) {
  465. self.buffer.as_mut()[field::HLEN] = value;
  466. }
  467. /// Sets the transaction ID.
  468. ///
  469. /// The transaction ID (called `xid` in the specification) is a random number used to
  470. /// associate messages and responses between client and server. The number is chosen by
  471. /// the client.
  472. pub fn set_transaction_id(&mut self, value: u32) {
  473. let field = &mut self.buffer.as_mut()[field::XID];
  474. NetworkEndian::write_u32(field, value)
  475. }
  476. /// Sets the ethernet address of the client.
  477. ///
  478. /// Sets the `chaddr` field.
  479. pub fn set_client_hardware_address(&mut self, value: EthernetAddress) {
  480. let field = &mut self.buffer.as_mut()[field::CHADDR];
  481. field.copy_from_slice(value.as_bytes());
  482. }
  483. /// Sets the hops field.
  484. ///
  485. /// The `hops` field is set to zero by clients and optionally used by relay agents.
  486. pub fn set_hops(&mut self, value: u8) {
  487. self.buffer.as_mut()[field::HOPS] = value;
  488. }
  489. /// Sets the `secs` field.
  490. ///
  491. /// The secs field is filled by clients and describes the number of seconds elapsed
  492. /// since client began process.
  493. pub fn set_secs(&mut self, value: u16) {
  494. let field = &mut self.buffer.as_mut()[field::SECS];
  495. NetworkEndian::write_u16(field, value);
  496. }
  497. /// Sets the value of the `magic cookie` field in the DHCP options.
  498. ///
  499. /// This field should be always be `0x63825363`.
  500. pub fn set_magic_number(&mut self, value: u32) {
  501. let field = &mut self.buffer.as_mut()[field::MAGIC_NUMBER];
  502. NetworkEndian::write_u32(field, value);
  503. }
  504. /// Sets the Ipv4 address of the client.
  505. ///
  506. /// This corresponds to the `ciaddr` field in the DHCP specification. According to it,
  507. /// this field is “only filled in if client is in `BOUND`, `RENEW` or `REBINDING` state
  508. /// and can respond to ARP requests”.
  509. pub fn set_client_ip(&mut self, value: Ipv4Address) {
  510. let field = &mut self.buffer.as_mut()[field::CIADDR];
  511. field.copy_from_slice(value.as_bytes());
  512. }
  513. /// Sets the value of the `yiaddr` field.
  514. pub fn set_your_ip(&mut self, value: Ipv4Address) {
  515. let field = &mut self.buffer.as_mut()[field::YIADDR];
  516. field.copy_from_slice(value.as_bytes());
  517. }
  518. /// Sets the value of the `siaddr` field.
  519. pub fn set_server_ip(&mut self, value: Ipv4Address) {
  520. let field = &mut self.buffer.as_mut()[field::SIADDR];
  521. field.copy_from_slice(value.as_bytes());
  522. }
  523. /// Sets the value of the `giaddr` field.
  524. pub fn set_relay_agent_ip(&mut self, value: Ipv4Address) {
  525. let field = &mut self.buffer.as_mut()[field::GIADDR];
  526. field.copy_from_slice(value.as_bytes());
  527. }
  528. /// Sets the flags to the specified value.
  529. pub fn set_flags(&mut self, val: Flags) {
  530. let field = &mut self.buffer.as_mut()[field::FLAGS];
  531. NetworkEndian::write_u16(field, val.bits());
  532. }
  533. }
  534. impl<'a, T: AsRef<[u8]> + AsMut<[u8]> + ?Sized> Packet<&'a mut T> {
  535. /// Return a pointer to the options.
  536. #[inline]
  537. pub fn options_mut(&mut self) -> Result<&mut [u8]> {
  538. let data = self.buffer.as_mut();
  539. data.get_mut(field::OPTIONS).ok_or(Error::Truncated)
  540. }
  541. }
  542. /// A high-level representation of a Dynamic Host Configuration Protocol packet.
  543. ///
  544. /// DHCP messages have the following layout (see [RFC 2131](https://tools.ietf.org/html/rfc2131)
  545. /// for details):
  546. ///
  547. /// ```no_rust
  548. /// 0 1 2 3
  549. /// 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
  550. /// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
  551. /// | message_type | htype (N/A) | hlen (N/A) | hops |
  552. /// +---------------+---------------+---------------+---------------+
  553. /// | transaction_id |
  554. /// +-------------------------------+-------------------------------+
  555. /// | secs | flags |
  556. /// +-------------------------------+-------------------------------+
  557. /// | client_ip |
  558. /// +---------------------------------------------------------------+
  559. /// | your_ip |
  560. /// +---------------------------------------------------------------+
  561. /// | server_ip |
  562. /// +---------------------------------------------------------------+
  563. /// | relay_agent_ip |
  564. /// +---------------------------------------------------------------+
  565. /// | |
  566. /// | client_hardware_address |
  567. /// | |
  568. /// | |
  569. /// +---------------------------------------------------------------+
  570. /// | |
  571. /// | sname (N/A) |
  572. /// +---------------------------------------------------------------+
  573. /// | |
  574. /// | file (N/A) |
  575. /// +---------------------------------------------------------------+
  576. /// | |
  577. /// | options |
  578. /// +---------------------------------------------------------------+
  579. /// ```
  580. ///
  581. /// It is assumed that the access layer is Ethernet, so `htype` (the field representing the
  582. /// hardware address type) is always set to `1`, and `hlen` (which represents the hardware address
  583. /// length) is set to `6`.
  584. ///
  585. /// The `options` field has a variable length.
  586. #[derive(Debug, PartialEq, Eq, Clone, Copy)]
  587. #[cfg_attr(feature = "defmt", derive(defmt::Format))]
  588. pub struct Repr<'a> {
  589. /// This field is also known as `op` in the RFC. It indicates the type of DHCP message this
  590. /// packet represents.
  591. pub message_type: MessageType,
  592. /// This field is also known as `xid` in the RFC. It is a random number chosen by the client,
  593. /// used by the client and server to associate messages and responses between a client and a
  594. /// server.
  595. pub transaction_id: u32,
  596. /// This field is also known as `chaddr` in the RFC and for networks where the access layer is
  597. /// ethernet, it is the client MAC address.
  598. pub client_hardware_address: EthernetAddress,
  599. /// This field is also known as `ciaddr` in the RFC. It is only filled in if client is in
  600. /// BOUND, RENEW or REBINDING state and can respond to ARP requests.
  601. pub client_ip: Ipv4Address,
  602. /// This field is also known as `yiaddr` in the RFC.
  603. pub your_ip: Ipv4Address,
  604. /// This field is also known as `siaddr` in the RFC. It may be set by the server in DHCPOFFER
  605. /// and DHCPACK messages, and represent the address of the next server to use in bootstrap.
  606. pub server_ip: Ipv4Address,
  607. /// Default gateway
  608. pub router: Option<Ipv4Address>,
  609. /// This field comes from a corresponding DhcpOption.
  610. pub subnet_mask: Option<Ipv4Address>,
  611. /// This field is also known as `giaddr` in the RFC. In order to allow DHCP clients on subnets
  612. /// not directly served by DHCP servers to communicate with DHCP servers, DHCP relay agents can
  613. /// be installed on these subnets. The DHCP client broadcasts on the local link; the relay
  614. /// agent receives the broadcast and transmits it to one or more DHCP servers using unicast.
  615. /// The relay agent stores its own IP address in the `relay_agent_ip` field of the DHCP packet.
  616. /// The DHCP server uses the `relay_agent_ip` to determine the subnet on which the relay agent
  617. /// received the broadcast, and allocates an IP address on that subnet. When the DHCP server
  618. /// replies to the client, it sends the reply to the `relay_agent_ip` address, again using
  619. /// unicast. The relay agent then retransmits the response on the local network
  620. pub relay_agent_ip: Ipv4Address,
  621. /// Broadcast flags. It can be set in DHCPDISCOVER, DHCPINFORM and DHCPREQUEST message if the
  622. /// client requires the response to be broadcasted.
  623. pub broadcast: bool,
  624. /// The "requested IP address" option. It can be used by clients in DHCPREQUEST or DHCPDISCOVER
  625. /// messages, or by servers in DHCPDECLINE messages.
  626. pub requested_ip: Option<Ipv4Address>,
  627. /// The "client identifier" option.
  628. ///
  629. /// The 'client identifier' is an opaque key, not to be interpreted by the server; for example,
  630. /// the 'client identifier' may contain a hardware address, identical to the contents of the
  631. /// 'chaddr' field, or it may contain another type of identifier, such as a DNS name. The
  632. /// 'client identifier' chosen by a DHCP client MUST be unique to that client within the subnet
  633. /// to which the client is attached. If the client uses a 'client identifier' in one message,
  634. /// it MUST use that same identifier in all subsequent messages, to ensure that all servers
  635. /// correctly identify the client.
  636. pub client_identifier: Option<EthernetAddress>,
  637. /// The "server identifier" option. It is used both to identify a DHCP server
  638. /// in a DHCP message and as a destination address from clients to servers.
  639. pub server_identifier: Option<Ipv4Address>,
  640. /// The parameter request list informs the server about which configuration parameters
  641. /// the client is interested in.
  642. pub parameter_request_list: Option<&'a [u8]>,
  643. /// DNS servers
  644. pub dns_servers: Option<[Option<Ipv4Address>; MAX_DNS_SERVER_COUNT]>,
  645. /// The maximum size dhcp packet the interface can receive
  646. pub max_size: Option<u16>,
  647. /// The DHCP IP lease duration, specified in seconds.
  648. pub lease_duration: Option<u32>,
  649. }
  650. impl<'a> Repr<'a> {
  651. /// Return the length of a packet that will be emitted from this high-level representation.
  652. pub fn buffer_len(&self) -> usize {
  653. let mut len = field::OPTIONS.start;
  654. // message type and end-of-options options
  655. len += 3 + 1;
  656. if self.requested_ip.is_some() {
  657. len += 6;
  658. }
  659. if self.client_identifier.is_some() {
  660. len += 9;
  661. }
  662. if self.server_identifier.is_some() {
  663. len += 6;
  664. }
  665. if self.max_size.is_some() {
  666. len += 4;
  667. }
  668. if self.router.is_some() {
  669. len += 6;
  670. }
  671. if self.subnet_mask.is_some() {
  672. len += 6;
  673. }
  674. if self.lease_duration.is_some() {
  675. len += 6;
  676. }
  677. if let Some(dns_servers) = self.dns_servers {
  678. len += 2;
  679. len += dns_servers.iter().flatten().count() * core::mem::size_of::<u32>();
  680. }
  681. if let Some(list) = self.parameter_request_list {
  682. len += list.len() + 2;
  683. }
  684. len
  685. }
  686. /// Parse a DHCP packet and return a high-level representation.
  687. pub fn parse<T>(packet: &Packet<&'a T>) -> Result<Self>
  688. where
  689. T: AsRef<[u8]> + ?Sized,
  690. {
  691. let transaction_id = packet.transaction_id();
  692. let client_hardware_address = packet.client_hardware_address();
  693. let client_ip = packet.client_ip();
  694. let your_ip = packet.your_ip();
  695. let server_ip = packet.server_ip();
  696. let relay_agent_ip = packet.relay_agent_ip();
  697. // only ethernet is supported right now
  698. match packet.hardware_type() {
  699. Hardware::Ethernet => {
  700. if packet.hardware_len() != 6 {
  701. return Err(Error::Malformed);
  702. }
  703. }
  704. Hardware::Unknown(_) => return Err(Error::Unrecognized), // unimplemented
  705. }
  706. if packet.magic_number() != DHCP_MAGIC_NUMBER {
  707. return Err(Error::Malformed);
  708. }
  709. let mut message_type = Err(Error::Malformed);
  710. let mut requested_ip = None;
  711. let mut client_identifier = None;
  712. let mut server_identifier = None;
  713. let mut router = None;
  714. let mut subnet_mask = None;
  715. let mut parameter_request_list = None;
  716. let mut dns_servers = None;
  717. let mut max_size = None;
  718. let mut lease_duration = None;
  719. let mut options = packet.options()?;
  720. while !options.is_empty() {
  721. let (next_options, option) = DhcpOption::parse(options)?;
  722. match option {
  723. DhcpOption::EndOfList => break,
  724. DhcpOption::Pad => {}
  725. DhcpOption::MessageType(value) => {
  726. if value.opcode() == packet.opcode() {
  727. message_type = Ok(value);
  728. }
  729. }
  730. DhcpOption::RequestedIp(ip) => {
  731. requested_ip = Some(ip);
  732. }
  733. DhcpOption::ClientIdentifier(eth_addr) => {
  734. client_identifier = Some(eth_addr);
  735. }
  736. DhcpOption::ServerIdentifier(ip) => {
  737. server_identifier = Some(ip);
  738. }
  739. DhcpOption::Router(ip) => {
  740. router = Some(ip);
  741. }
  742. DhcpOption::SubnetMask(mask) => {
  743. subnet_mask = Some(mask);
  744. }
  745. DhcpOption::MaximumDhcpMessageSize(size) => {
  746. max_size = Some(size);
  747. }
  748. DhcpOption::IpLeaseTime(duration) => {
  749. lease_duration = Some(duration);
  750. }
  751. DhcpOption::Other {
  752. kind: field::OPT_PARAMETER_REQUEST_LIST,
  753. data,
  754. } => {
  755. parameter_request_list = Some(data);
  756. }
  757. DhcpOption::Other {
  758. kind: field::OPT_DOMAIN_NAME_SERVER,
  759. data,
  760. } => {
  761. let mut servers = [None; MAX_DNS_SERVER_COUNT];
  762. let chunk_size = 4;
  763. for (server, chunk) in servers.iter_mut().zip(data.chunks(chunk_size)) {
  764. if chunk.len() != chunk_size {
  765. return Err(Error::Malformed);
  766. }
  767. *server = Some(Ipv4Address::from_bytes(chunk));
  768. }
  769. dns_servers = Some(servers);
  770. }
  771. DhcpOption::Other { .. } => {}
  772. }
  773. options = next_options;
  774. }
  775. let broadcast = packet.flags().contains(Flags::BROADCAST);
  776. Ok(Repr {
  777. transaction_id,
  778. client_hardware_address,
  779. client_ip,
  780. your_ip,
  781. server_ip,
  782. relay_agent_ip,
  783. broadcast,
  784. requested_ip,
  785. server_identifier,
  786. router,
  787. subnet_mask,
  788. client_identifier,
  789. parameter_request_list,
  790. dns_servers,
  791. max_size,
  792. lease_duration,
  793. message_type: message_type?,
  794. })
  795. }
  796. /// Emit a high-level representation into a Dynamic Host
  797. /// Configuration Protocol packet.
  798. pub fn emit<T>(&self, packet: &mut Packet<&mut T>) -> Result<()>
  799. where
  800. T: AsRef<[u8]> + AsMut<[u8]> + ?Sized,
  801. {
  802. packet.set_sname_and_boot_file_to_zero();
  803. packet.set_opcode(self.message_type.opcode());
  804. packet.set_hardware_type(Hardware::Ethernet);
  805. packet.set_hardware_len(6);
  806. packet.set_transaction_id(self.transaction_id);
  807. packet.set_client_hardware_address(self.client_hardware_address);
  808. packet.set_hops(0);
  809. packet.set_secs(0); // TODO
  810. packet.set_magic_number(0x63825363);
  811. packet.set_client_ip(self.client_ip);
  812. packet.set_your_ip(self.your_ip);
  813. packet.set_server_ip(self.server_ip);
  814. packet.set_relay_agent_ip(self.relay_agent_ip);
  815. let mut flags = Flags::empty();
  816. if self.broadcast {
  817. flags |= Flags::BROADCAST;
  818. }
  819. packet.set_flags(flags);
  820. {
  821. let mut options = packet.options_mut()?;
  822. options = DhcpOption::MessageType(self.message_type).emit(options);
  823. if let Some(eth_addr) = self.client_identifier {
  824. options = DhcpOption::ClientIdentifier(eth_addr).emit(options);
  825. }
  826. if let Some(ip) = self.server_identifier {
  827. options = DhcpOption::ServerIdentifier(ip).emit(options);
  828. }
  829. if let Some(ip) = self.router {
  830. options = DhcpOption::Router(ip).emit(options);
  831. }
  832. if let Some(ip) = self.subnet_mask {
  833. options = DhcpOption::SubnetMask(ip).emit(options);
  834. }
  835. if let Some(ip) = self.requested_ip {
  836. options = DhcpOption::RequestedIp(ip).emit(options);
  837. }
  838. if let Some(size) = self.max_size {
  839. options = DhcpOption::MaximumDhcpMessageSize(size).emit(options);
  840. }
  841. if let Some(duration) = self.lease_duration {
  842. options = DhcpOption::IpLeaseTime(duration).emit(options);
  843. }
  844. if let Some(dns_servers) = self.dns_servers {
  845. const IP_SIZE: usize = core::mem::size_of::<u32>();
  846. let mut servers = [0; MAX_DNS_SERVER_COUNT * IP_SIZE];
  847. let data_len = dns_servers
  848. .iter()
  849. .flatten()
  850. .enumerate()
  851. .inspect(|(i, ip)| {
  852. servers[(i * IP_SIZE)..((i + 1) * IP_SIZE)].copy_from_slice(ip.as_bytes());
  853. })
  854. .count()
  855. * IP_SIZE;
  856. let option = DhcpOption::Other {
  857. kind: field::OPT_DOMAIN_NAME_SERVER,
  858. data: &servers[..data_len],
  859. };
  860. options = option.emit(options);
  861. }
  862. if let Some(list) = self.parameter_request_list {
  863. options = DhcpOption::Other {
  864. kind: field::OPT_PARAMETER_REQUEST_LIST,
  865. data: list,
  866. }
  867. .emit(options);
  868. }
  869. DhcpOption::EndOfList.emit(options);
  870. }
  871. Ok(())
  872. }
  873. }
  874. #[cfg(test)]
  875. mod test {
  876. use super::*;
  877. use crate::wire::Ipv4Address;
  878. const MAGIC_COOKIE: u32 = 0x63825363;
  879. static DISCOVER_BYTES: &[u8] = &[
  880. 0x01, 0x01, 0x06, 0x00, 0x00, 0x00, 0x3d, 0x1d, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  881. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0b,
  882. 0x82, 0x01, 0xfc, 0x42, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  883. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  884. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  885. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  886. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  887. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  888. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  889. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  890. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  891. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  892. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  893. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  894. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  895. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x63, 0x82, 0x53, 0x63,
  896. 0x35, 0x01, 0x01, 0x3d, 0x07, 0x01, 0x00, 0x0b, 0x82, 0x01, 0xfc, 0x42, 0x32, 0x04, 0x00,
  897. 0x00, 0x00, 0x00, 0x39, 0x2, 0x5, 0xdc, 0x37, 0x04, 0x01, 0x03, 0x06, 0x2a, 0xff, 0x00,
  898. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  899. ];
  900. static ACK_DNS_SERVER_BYTES: &[u8] = &[
  901. 0x02, 0x01, 0x06, 0x00, 0xcc, 0x34, 0x75, 0xab, 0x00, 0x00, 0x80, 0x00, 0x0a, 0xff, 0x06,
  902. 0x91, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0a, 0xff, 0x06, 0xfe, 0x34, 0x17,
  903. 0xeb, 0xc9, 0xaa, 0x2f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  904. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  905. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  906. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  907. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  908. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  909. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  910. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  911. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  912. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  913. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  914. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  915. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  916. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x63, 0x82, 0x53, 0x63,
  917. 0x35, 0x01, 0x05, 0x36, 0x04, 0xa3, 0x01, 0x4a, 0x16, 0x01, 0x04, 0xff, 0xff, 0xff, 0x00,
  918. 0x2b, 0x05, 0xdc, 0x03, 0x4e, 0x41, 0x50, 0x0f, 0x15, 0x6e, 0x61, 0x74, 0x2e, 0x70, 0x68,
  919. 0x79, 0x73, 0x69, 0x63, 0x73, 0x2e, 0x6f, 0x78, 0x2e, 0x61, 0x63, 0x2e, 0x75, 0x6b, 0x00,
  920. 0x03, 0x04, 0x0a, 0xff, 0x06, 0xfe, 0x06, 0x10, 0xa3, 0x01, 0x4a, 0x06, 0xa3, 0x01, 0x4a,
  921. 0x07, 0xa3, 0x01, 0x4a, 0x03, 0xa3, 0x01, 0x4a, 0x04, 0x2c, 0x10, 0xa3, 0x01, 0x4a, 0x03,
  922. 0xa3, 0x01, 0x4a, 0x04, 0xa3, 0x01, 0x4a, 0x06, 0xa3, 0x01, 0x4a, 0x07, 0x2e, 0x01, 0x08,
  923. 0xff,
  924. ];
  925. static ACK_LEASE_TIME_BYTES: &[u8] = &[
  926. 0x02, 0x01, 0x06, 0x00, 0x00, 0x00, 0x00, 0x06, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  927. 0x00, 0x0a, 0x22, 0x10, 0x0b, 0x0a, 0x22, 0x10, 0x0a, 0x00, 0x00, 0x00, 0x00, 0x04, 0x91,
  928. 0x62, 0xd2, 0xa8, 0x6f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  929. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  930. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  931. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  932. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  933. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  934. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  935. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  936. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  937. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  938. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  939. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  940. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  941. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x63, 0x82, 0x53, 0x63,
  942. 0x35, 0x01, 0x05, 0x36, 0x04, 0x0a, 0x22, 0x10, 0x0a, 0x33, 0x04, 0x00, 0x00, 0x02, 0x56,
  943. 0x01, 0x04, 0xff, 0xff, 0xff, 0x00, 0x03, 0x04, 0x0a, 0x22, 0x10, 0x0a, 0xff, 0x00, 0x00,
  944. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  945. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  946. ];
  947. const IP_NULL: Ipv4Address = Ipv4Address([0, 0, 0, 0]);
  948. const CLIENT_MAC: EthernetAddress = EthernetAddress([0x0, 0x0b, 0x82, 0x01, 0xfc, 0x42]);
  949. const DHCP_SIZE: u16 = 1500;
  950. #[test]
  951. fn test_deconstruct_discover() {
  952. let packet = Packet::new_unchecked(DISCOVER_BYTES);
  953. assert_eq!(packet.magic_number(), MAGIC_COOKIE);
  954. assert_eq!(packet.opcode(), OpCode::Request);
  955. assert_eq!(packet.hardware_type(), Hardware::Ethernet);
  956. assert_eq!(packet.hardware_len(), 6);
  957. assert_eq!(packet.hops(), 0);
  958. assert_eq!(packet.transaction_id(), 0x3d1d);
  959. assert_eq!(packet.secs(), 0);
  960. assert_eq!(packet.client_ip(), IP_NULL);
  961. assert_eq!(packet.your_ip(), IP_NULL);
  962. assert_eq!(packet.server_ip(), IP_NULL);
  963. assert_eq!(packet.relay_agent_ip(), IP_NULL);
  964. assert_eq!(packet.client_hardware_address(), CLIENT_MAC);
  965. let options = packet.options().unwrap();
  966. assert_eq!(options.len(), 3 + 9 + 6 + 4 + 6 + 1 + 7);
  967. let (options, message_type) = DhcpOption::parse(options).unwrap();
  968. assert_eq!(message_type, DhcpOption::MessageType(MessageType::Discover));
  969. assert_eq!(options.len(), 9 + 6 + 4 + 6 + 1 + 7);
  970. let (options, client_id) = DhcpOption::parse(options).unwrap();
  971. assert_eq!(client_id, DhcpOption::ClientIdentifier(CLIENT_MAC));
  972. assert_eq!(options.len(), 6 + 4 + 6 + 1 + 7);
  973. let (options, client_id) = DhcpOption::parse(options).unwrap();
  974. assert_eq!(client_id, DhcpOption::RequestedIp(IP_NULL));
  975. assert_eq!(options.len(), 4 + 6 + 1 + 7);
  976. let (options, msg_size) = DhcpOption::parse(options).unwrap();
  977. assert_eq!(msg_size, DhcpOption::MaximumDhcpMessageSize(DHCP_SIZE));
  978. assert_eq!(options.len(), 6 + 1 + 7);
  979. let (options, client_id) = DhcpOption::parse(options).unwrap();
  980. assert_eq!(
  981. client_id,
  982. DhcpOption::Other {
  983. kind: field::OPT_PARAMETER_REQUEST_LIST,
  984. data: &[1, 3, 6, 42]
  985. }
  986. );
  987. assert_eq!(options.len(), 1 + 7);
  988. let (options, client_id) = DhcpOption::parse(options).unwrap();
  989. assert_eq!(client_id, DhcpOption::EndOfList);
  990. assert_eq!(options.len(), 7); // padding
  991. }
  992. #[test]
  993. fn test_construct_discover() {
  994. let mut bytes = vec![0xa5; 276];
  995. let mut packet = Packet::new_unchecked(&mut bytes);
  996. packet.set_magic_number(MAGIC_COOKIE);
  997. packet.set_sname_and_boot_file_to_zero();
  998. packet.set_opcode(OpCode::Request);
  999. packet.set_hardware_type(Hardware::Ethernet);
  1000. packet.set_hardware_len(6);
  1001. packet.set_hops(0);
  1002. packet.set_transaction_id(0x3d1d);
  1003. packet.set_secs(0);
  1004. packet.set_flags(Flags::empty());
  1005. packet.set_client_ip(IP_NULL);
  1006. packet.set_your_ip(IP_NULL);
  1007. packet.set_server_ip(IP_NULL);
  1008. packet.set_relay_agent_ip(IP_NULL);
  1009. packet.set_client_hardware_address(CLIENT_MAC);
  1010. {
  1011. let mut options = packet.options_mut().unwrap();
  1012. options = DhcpOption::MessageType(MessageType::Discover).emit(options);
  1013. options = DhcpOption::ClientIdentifier(CLIENT_MAC).emit(options);
  1014. options = DhcpOption::RequestedIp(IP_NULL).emit(options);
  1015. options = DhcpOption::MaximumDhcpMessageSize(DHCP_SIZE).emit(options);
  1016. let option = DhcpOption::Other {
  1017. kind: field::OPT_PARAMETER_REQUEST_LIST,
  1018. data: &[1, 3, 6, 42],
  1019. };
  1020. options = option.emit(options);
  1021. DhcpOption::EndOfList.emit(options);
  1022. }
  1023. let packet = &mut packet.into_inner()[..];
  1024. for byte in &mut packet[269..276] {
  1025. *byte = 0; // padding bytes
  1026. }
  1027. assert_eq!(packet, DISCOVER_BYTES);
  1028. }
  1029. fn offer_repr() -> Repr<'static> {
  1030. Repr {
  1031. message_type: MessageType::Offer,
  1032. transaction_id: 0x3d1d,
  1033. client_hardware_address: CLIENT_MAC,
  1034. client_ip: IP_NULL,
  1035. your_ip: IP_NULL,
  1036. server_ip: IP_NULL,
  1037. router: Some(IP_NULL),
  1038. subnet_mask: Some(IP_NULL),
  1039. relay_agent_ip: IP_NULL,
  1040. broadcast: false,
  1041. requested_ip: None,
  1042. client_identifier: Some(CLIENT_MAC),
  1043. server_identifier: None,
  1044. parameter_request_list: None,
  1045. dns_servers: None,
  1046. max_size: None,
  1047. lease_duration: Some(0xffff_ffff), // Infinite lease
  1048. }
  1049. }
  1050. fn discover_repr() -> Repr<'static> {
  1051. Repr {
  1052. message_type: MessageType::Discover,
  1053. transaction_id: 0x3d1d,
  1054. client_hardware_address: CLIENT_MAC,
  1055. client_ip: IP_NULL,
  1056. your_ip: IP_NULL,
  1057. server_ip: IP_NULL,
  1058. router: None,
  1059. subnet_mask: None,
  1060. relay_agent_ip: IP_NULL,
  1061. broadcast: false,
  1062. max_size: Some(DHCP_SIZE),
  1063. lease_duration: None,
  1064. requested_ip: Some(IP_NULL),
  1065. client_identifier: Some(CLIENT_MAC),
  1066. server_identifier: None,
  1067. parameter_request_list: Some(&[1, 3, 6, 42]),
  1068. dns_servers: None,
  1069. }
  1070. }
  1071. #[test]
  1072. fn test_parse_discover() {
  1073. let packet = Packet::new_unchecked(DISCOVER_BYTES);
  1074. let repr = Repr::parse(&packet).unwrap();
  1075. assert_eq!(repr, discover_repr());
  1076. }
  1077. #[test]
  1078. fn test_emit_discover() {
  1079. let repr = discover_repr();
  1080. let mut bytes = vec![0xa5; repr.buffer_len()];
  1081. let mut packet = Packet::new_unchecked(&mut bytes);
  1082. repr.emit(&mut packet).unwrap();
  1083. let packet = &packet.into_inner()[..];
  1084. let packet_len = packet.len();
  1085. assert_eq!(packet, &DISCOVER_BYTES[..packet_len]);
  1086. for byte in &DISCOVER_BYTES[packet_len..] {
  1087. assert_eq!(*byte, 0); // padding bytes
  1088. }
  1089. }
  1090. #[test]
  1091. fn test_emit_offer() {
  1092. let repr = offer_repr();
  1093. let mut bytes = vec![0xa5; repr.buffer_len()];
  1094. let mut packet = Packet::new_unchecked(&mut bytes);
  1095. repr.emit(&mut packet).unwrap();
  1096. }
  1097. #[test]
  1098. fn test_emit_offer_dns() {
  1099. let repr = {
  1100. let mut repr = offer_repr();
  1101. repr.dns_servers = Some([
  1102. Some(Ipv4Address([163, 1, 74, 6])),
  1103. Some(Ipv4Address([163, 1, 74, 7])),
  1104. Some(Ipv4Address([163, 1, 74, 3])),
  1105. ]);
  1106. repr
  1107. };
  1108. let mut bytes = vec![0xa5; repr.buffer_len()];
  1109. let mut packet = Packet::new_unchecked(&mut bytes);
  1110. repr.emit(&mut packet).unwrap();
  1111. let packet = Packet::new_unchecked(&bytes);
  1112. let repr_parsed = Repr::parse(&packet).unwrap();
  1113. assert_eq!(
  1114. repr_parsed.dns_servers,
  1115. Some([
  1116. Some(Ipv4Address([163, 1, 74, 6])),
  1117. Some(Ipv4Address([163, 1, 74, 7])),
  1118. Some(Ipv4Address([163, 1, 74, 3]))
  1119. ])
  1120. );
  1121. }
  1122. #[test]
  1123. fn test_emit_dhcp_option() {
  1124. static DATA: &[u8] = &[1, 3, 6];
  1125. let mut bytes = vec![0xa5; 5];
  1126. let dhcp_option = DhcpOption::Other {
  1127. kind: field::OPT_PARAMETER_REQUEST_LIST,
  1128. data: DATA,
  1129. };
  1130. {
  1131. let rest = dhcp_option.emit(&mut bytes);
  1132. assert_eq!(rest.len(), 0);
  1133. }
  1134. assert_eq!(
  1135. &bytes[0..2],
  1136. &[field::OPT_PARAMETER_REQUEST_LIST, DATA.len() as u8]
  1137. );
  1138. assert_eq!(&bytes[2..], DATA);
  1139. }
  1140. #[test]
  1141. fn test_parse_ack_dns_servers() {
  1142. let packet = Packet::new_unchecked(ACK_DNS_SERVER_BYTES);
  1143. let repr = Repr::parse(&packet).unwrap();
  1144. // The packet described by ACK_BYTES advertises 4 DNS servers
  1145. // Here we ensure that we correctly parse the first 3 into our fixed
  1146. // length-3 array (see issue #305)
  1147. assert_eq!(
  1148. repr.dns_servers,
  1149. Some([
  1150. Some(Ipv4Address([163, 1, 74, 6])),
  1151. Some(Ipv4Address([163, 1, 74, 7])),
  1152. Some(Ipv4Address([163, 1, 74, 3]))
  1153. ])
  1154. );
  1155. }
  1156. #[test]
  1157. fn test_parse_ack_lease_duration() {
  1158. let packet = Packet::new_unchecked(ACK_LEASE_TIME_BYTES);
  1159. let repr = Repr::parse(&packet).unwrap();
  1160. // Verify that the lease time in the ACK is properly parsed. The packet contains a lease
  1161. // duration of 598s.
  1162. assert_eq!(repr.lease_duration, Some(598));
  1163. }
  1164. }