dhcp_client.rs 3.7 KB

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