dhcp_client.rs 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. #![allow(clippy::option_map_unit_fn)]
  2. mod utils;
  3. use log::*;
  4. use std::collections::BTreeMap;
  5. use std::os::unix::io::AsRawFd;
  6. use smoltcp::iface::{Interface, InterfaceBuilder, NeighborCache, Routes, SocketSet};
  7. use smoltcp::socket::dhcpv4;
  8. use smoltcp::time::Instant;
  9. use smoltcp::wire::{EthernetAddress, IpCidr, Ipv4Address, Ipv4Cidr};
  10. use smoltcp::{
  11. phy::{wait as phy_wait, Device, Medium},
  12. time::Duration,
  13. };
  14. fn main() {
  15. #[cfg(feature = "log")]
  16. utils::setup_logging("");
  17. let (mut opts, mut free) = utils::create_options();
  18. utils::add_tuntap_options(&mut opts, &mut free);
  19. utils::add_middleware_options(&mut opts, &mut free);
  20. let mut matches = utils::parse_options(&opts, free);
  21. let device = utils::parse_tuntap_options(&mut matches);
  22. let fd = device.as_raw_fd();
  23. let mut device =
  24. utils::parse_middleware_options(&mut matches, device, /*loopback=*/ false);
  25. let neighbor_cache = NeighborCache::new(BTreeMap::new());
  26. let ethernet_addr = EthernetAddress([0x02, 0x00, 0x00, 0x00, 0x00, 0x01]);
  27. let mut ip_addrs = heapless::Vec::<IpCidr, 5>::new();
  28. ip_addrs
  29. .push(IpCidr::new(Ipv4Address::UNSPECIFIED.into(), 0))
  30. .unwrap();
  31. let mut routes_storage = [None; 1];
  32. let routes = Routes::new(&mut routes_storage[..]);
  33. let medium = device.capabilities().medium;
  34. let mut builder = InterfaceBuilder::new().ip_addrs(ip_addrs).routes(routes);
  35. if medium == Medium::Ethernet {
  36. builder = builder
  37. .hardware_addr(ethernet_addr.into())
  38. .neighbor_cache(neighbor_cache);
  39. }
  40. let mut iface = builder.finalize(&mut device);
  41. let mut dhcp_socket = dhcpv4::Socket::new();
  42. // Set a ridiculously short max lease time to show DHCP renews work properly.
  43. // This will cause the DHCP client to start renewing after 5 seconds, and give up the
  44. // lease after 10 seconds if renew hasn't succeeded.
  45. // IMPORTANT: This should be removed in production.
  46. dhcp_socket.set_max_lease_duration(Some(Duration::from_secs(10)));
  47. let mut sockets = SocketSet::new(vec![]);
  48. let dhcp_handle = sockets.add(dhcp_socket);
  49. loop {
  50. let timestamp = Instant::now();
  51. if let Err(e) = iface.poll(timestamp, &mut device, &mut sockets) {
  52. debug!("poll error: {}", e);
  53. }
  54. let event = sockets.get_mut::<dhcpv4::Socket>(dhcp_handle).poll();
  55. match event {
  56. None => {}
  57. Some(dhcpv4::Event::Configured(config)) => {
  58. debug!("DHCP config acquired!");
  59. debug!("IP address: {}", config.address);
  60. set_ipv4_addr(&mut iface, config.address);
  61. if let Some(router) = config.router {
  62. debug!("Default gateway: {}", router);
  63. iface.routes_mut().add_default_ipv4_route(router).unwrap();
  64. } else {
  65. debug!("Default gateway: None");
  66. iface.routes_mut().remove_default_ipv4_route();
  67. }
  68. for (i, s) in config.dns_servers.iter().enumerate() {
  69. debug!("DNS server {}: {}", i, s);
  70. }
  71. }
  72. Some(dhcpv4::Event::Deconfigured) => {
  73. debug!("DHCP lost config!");
  74. set_ipv4_addr(&mut iface, Ipv4Cidr::new(Ipv4Address::UNSPECIFIED, 0));
  75. iface.routes_mut().remove_default_ipv4_route();
  76. }
  77. }
  78. phy_wait(fd, iface.poll_delay(timestamp, &sockets)).expect("wait error");
  79. }
  80. }
  81. fn set_ipv4_addr(iface: &mut Interface<'_>, cidr: Ipv4Cidr) {
  82. iface.update_ip_addrs(|addrs| {
  83. let dest = addrs.iter_mut().next().unwrap();
  84. *dest = IpCidr::Ipv4(cidr);
  85. });
  86. }