|
@@ -3,7 +3,7 @@ use core::task::Waker;
|
|
|
|
|
|
use crate::iface::Context;
|
|
|
use crate::time::{Duration, Instant};
|
|
|
-use crate::wire::dhcpv4::field as dhcpv4_field;
|
|
|
+use crate::wire::dhcpv4::{field as dhcpv4_field, DhcpOptionsBuffer};
|
|
|
use crate::wire::HardwareAddress;
|
|
|
use crate::wire::{
|
|
|
DhcpMessageType, DhcpPacket, DhcpRepr, IpAddress, IpProtocol, Ipv4Address, Ipv4Cidr, Ipv4Repr,
|
|
@@ -15,18 +15,9 @@ use super::WakerRegistration;
|
|
|
|
|
|
use super::PollAt;
|
|
|
|
|
|
-const DISCOVER_TIMEOUT: Duration = Duration::from_secs(10);
|
|
|
-
|
|
|
-// timeout doubles every 2 tries.
|
|
|
-// total time 5 + 5 + 10 + 10 + 20 = 50s
|
|
|
-const REQUEST_TIMEOUT: Duration = Duration::from_secs(5);
|
|
|
-const REQUEST_RETRIES: u16 = 5;
|
|
|
-
|
|
|
-const MIN_RENEW_TIMEOUT: Duration = Duration::from_secs(60);
|
|
|
-
|
|
|
const DEFAULT_LEASE_DURATION: Duration = Duration::from_secs(120);
|
|
|
|
|
|
-const PARAMETER_REQUEST_LIST: &[u8] = &[
|
|
|
+const DEFAULT_PARAMETER_REQUEST_LIST: &[u8] = &[
|
|
|
dhcpv4_field::OPT_SUBNET_MASK,
|
|
|
dhcpv4_field::OPT_ROUTER,
|
|
|
dhcpv4_field::OPT_DOMAIN_NAME_SERVER,
|
|
@@ -35,7 +26,10 @@ const PARAMETER_REQUEST_LIST: &[u8] = &[
|
|
|
/// IPv4 configuration data provided by the DHCP server.
|
|
|
#[derive(Debug, Eq, PartialEq, Clone, Copy)]
|
|
|
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
|
|
-pub struct Config {
|
|
|
+pub struct Config<'a> {
|
|
|
+ /// Information on how to reach the DHCP server that responded with DHCP
|
|
|
+ /// configuration.
|
|
|
+ pub server: ServerInfo,
|
|
|
/// IP address
|
|
|
pub address: Ipv4Cidr,
|
|
|
/// Router address, also known as default gateway. Does not necessarily
|
|
@@ -43,17 +37,19 @@ pub struct Config {
|
|
|
pub router: Option<Ipv4Address>,
|
|
|
/// DNS servers
|
|
|
pub dns_servers: [Option<Ipv4Address>; DHCP_MAX_DNS_SERVER_COUNT],
|
|
|
+ /// Received DHCP packet
|
|
|
+ pub packet: Option<DhcpPacket<&'a [u8]>>,
|
|
|
}
|
|
|
|
|
|
/// Information on how to reach a DHCP server.
|
|
|
-#[derive(Debug, Clone, Copy)]
|
|
|
+#[derive(Debug, Clone, Copy, Eq, PartialEq)]
|
|
|
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
|
|
-struct ServerInfo {
|
|
|
+pub struct ServerInfo {
|
|
|
/// IP address to use as destination in outgoing packets
|
|
|
- address: Ipv4Address,
|
|
|
+ pub address: Ipv4Address,
|
|
|
/// Server identifier to use in outgoing packets. Usually equal to server_address,
|
|
|
/// but may differ in some situations (eg DHCP relays)
|
|
|
- identifier: Ipv4Address,
|
|
|
+ pub identifier: Ipv4Address,
|
|
|
}
|
|
|
|
|
|
#[derive(Debug)]
|
|
@@ -79,10 +75,8 @@ struct RequestState {
|
|
|
#[derive(Debug)]
|
|
|
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
|
|
struct RenewState {
|
|
|
- /// Server that gave us the lease
|
|
|
- server: ServerInfo,
|
|
|
/// Active network config
|
|
|
- config: Config,
|
|
|
+ config: Config<'static>,
|
|
|
|
|
|
/// Renew timer. When reached, we will start attempting
|
|
|
/// to renew this lease with the DHCP server.
|
|
@@ -104,18 +98,40 @@ enum ClientState {
|
|
|
Renewing(RenewState),
|
|
|
}
|
|
|
|
|
|
+/// Timeout and retry configuration.
|
|
|
+#[derive(Debug, PartialEq, Eq, Copy, Clone)]
|
|
|
+#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
|
|
+pub struct RetryConfig {
|
|
|
+ pub discover_timeout: Duration,
|
|
|
+ /// The REQUEST timeout doubles every 2 tries.
|
|
|
+ pub initial_request_timeout: Duration,
|
|
|
+ pub request_retries: u16,
|
|
|
+ pub min_renew_timeout: Duration,
|
|
|
+}
|
|
|
+
|
|
|
+impl Default for RetryConfig {
|
|
|
+ fn default() -> Self {
|
|
|
+ Self {
|
|
|
+ discover_timeout: Duration::from_secs(10),
|
|
|
+ initial_request_timeout: Duration::from_secs(5),
|
|
|
+ request_retries: 5,
|
|
|
+ min_renew_timeout: Duration::from_secs(60),
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
/// Return value for the `Dhcpv4Socket::poll` function
|
|
|
#[derive(Debug, PartialEq, Eq)]
|
|
|
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
|
|
-pub enum Event {
|
|
|
+pub enum Event<'a> {
|
|
|
/// Configuration has been lost (for example, the lease has expired)
|
|
|
Deconfigured,
|
|
|
/// Configuration has been newly acquired, or modified.
|
|
|
- Configured(Config),
|
|
|
+ Configured(Config<'a>),
|
|
|
}
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
-pub struct Socket {
|
|
|
+pub struct Socket<'a> {
|
|
|
/// State of the DHCP client.
|
|
|
state: ClientState,
|
|
|
/// Set to true on config/state change, cleared back to false by the `config` function.
|
|
@@ -127,9 +143,20 @@ pub struct Socket {
|
|
|
/// Useful to react faster to IP configuration changes and to test whether renews work correctly.
|
|
|
max_lease_duration: Option<Duration>,
|
|
|
|
|
|
+ retry_config: RetryConfig,
|
|
|
+
|
|
|
/// Ignore NAKs.
|
|
|
ignore_naks: bool,
|
|
|
|
|
|
+ /// A buffer contains options additional to be added to outgoing DHCP
|
|
|
+ /// packets.
|
|
|
+ outgoing_options: Option<DhcpOptionsBuffer<&'a [u8]>>,
|
|
|
+ /// A buffer containing all requested parameters.
|
|
|
+ parameter_request_list: Option<&'a [u8]>,
|
|
|
+
|
|
|
+ /// Incoming DHCP packets are copied into this buffer, overwriting the previous.
|
|
|
+ receive_packet_buffer: Option<&'a mut [u8]>,
|
|
|
+
|
|
|
/// Waker registration
|
|
|
#[cfg(feature = "async")]
|
|
|
waker: WakerRegistration,
|
|
@@ -140,7 +167,7 @@ pub struct Socket {
|
|
|
/// The socket acquires an IP address configuration through DHCP autonomously.
|
|
|
/// You must query the configuration with `.poll()` after every call to `Interface::poll()`,
|
|
|
/// and apply the configuration to the `Interface`.
|
|
|
-impl Socket {
|
|
|
+impl<'a> Socket<'a> {
|
|
|
/// Create a DHCPv4 socket
|
|
|
#[allow(clippy::new_without_default)]
|
|
|
pub fn new() -> Self {
|
|
@@ -151,12 +178,39 @@ impl Socket {
|
|
|
config_changed: true,
|
|
|
transaction_id: 1,
|
|
|
max_lease_duration: None,
|
|
|
+ retry_config: RetryConfig::default(),
|
|
|
ignore_naks: false,
|
|
|
+ outgoing_options: None,
|
|
|
+ parameter_request_list: None,
|
|
|
+ receive_packet_buffer: None,
|
|
|
#[cfg(feature = "async")]
|
|
|
waker: WakerRegistration::new(),
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+ /// Set the retry/timeouts configuration.
|
|
|
+ pub fn set_retry_config(&mut self, config: RetryConfig) {
|
|
|
+ self.retry_config = config;
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Set the outgoing options buffer.
|
|
|
+ pub fn set_outgoing_options_buffer(&mut self, options_buffer: DhcpOptionsBuffer<&'a [u8]>) {
|
|
|
+ self.outgoing_options = Some(options_buffer);
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Set the buffer into which incoming DHCP packets are copied into.
|
|
|
+ pub fn set_receive_packet_buffer(&mut self, buffer: &'a mut [u8]) {
|
|
|
+ self.receive_packet_buffer = Some(buffer);
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Set the parameter request list.
|
|
|
+ ///
|
|
|
+ /// This should contain at least `OPT_SUBNET_MASK` (`1`), `OPT_ROUTER`
|
|
|
+ /// (`3`), and `OPT_DOMAIN_NAME_SERVER` (`6`).
|
|
|
+ pub fn set_parameter_request_list(&mut self, parameter_request_list: &'a [u8]) {
|
|
|
+ self.parameter_request_list = Some(parameter_request_list);
|
|
|
+ }
|
|
|
+
|
|
|
/// Get the configured max lease duration.
|
|
|
///
|
|
|
/// See also [`Self::set_max_lease_duration()`]
|
|
@@ -256,6 +310,13 @@ impl Socket {
|
|
|
dhcp_repr
|
|
|
);
|
|
|
|
|
|
+ // Copy over the payload into the receive packet buffer.
|
|
|
+ if let Some(buffer) = self.receive_packet_buffer.as_mut() {
|
|
|
+ if let Some(buffer) = buffer.get_mut(..payload.len()) {
|
|
|
+ buffer.copy_from_slice(payload);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
match (&mut self.state, dhcp_repr.message_type) {
|
|
|
(ClientState::Discovering(_state), DhcpMessageType::Offer) => {
|
|
|
if !dhcp_repr.your_ip.is_unicast() {
|
|
@@ -275,10 +336,9 @@ impl Socket {
|
|
|
}
|
|
|
(ClientState::Requesting(state), DhcpMessageType::Ack) => {
|
|
|
if let Some((config, renew_at, expires_at)) =
|
|
|
- Self::parse_ack(cx.now(), &dhcp_repr, self.max_lease_duration)
|
|
|
+ Self::parse_ack(cx.now(), &dhcp_repr, self.max_lease_duration, state.server)
|
|
|
{
|
|
|
self.state = ClientState::Renewing(RenewState {
|
|
|
- server: state.server,
|
|
|
config,
|
|
|
renew_at,
|
|
|
expires_at,
|
|
@@ -292,9 +352,12 @@ impl Socket {
|
|
|
}
|
|
|
}
|
|
|
(ClientState::Renewing(state), DhcpMessageType::Ack) => {
|
|
|
- if let Some((config, renew_at, expires_at)) =
|
|
|
- Self::parse_ack(cx.now(), &dhcp_repr, self.max_lease_duration)
|
|
|
- {
|
|
|
+ if let Some((config, renew_at, expires_at)) = Self::parse_ack(
|
|
|
+ cx.now(),
|
|
|
+ &dhcp_repr,
|
|
|
+ self.max_lease_duration,
|
|
|
+ state.config.server,
|
|
|
+ ) {
|
|
|
state.renew_at = renew_at;
|
|
|
state.expires_at = expires_at;
|
|
|
if state.config != config {
|
|
@@ -321,7 +384,8 @@ impl Socket {
|
|
|
now: Instant,
|
|
|
dhcp_repr: &DhcpRepr,
|
|
|
max_lease_duration: Option<Duration>,
|
|
|
- ) -> Option<(Config, Instant, Instant)> {
|
|
|
+ server: ServerInfo,
|
|
|
+ ) -> Option<(Config<'static>, Instant, Instant)> {
|
|
|
let subnet_mask = match dhcp_repr.subnet_mask {
|
|
|
Some(subnet_mask) => subnet_mask,
|
|
|
None => {
|
|
@@ -365,9 +429,11 @@ impl Socket {
|
|
|
}
|
|
|
}
|
|
|
let config = Config {
|
|
|
+ server,
|
|
|
address: Ipv4Cidr::new(dhcp_repr.your_ip, prefix_len),
|
|
|
router: dhcp_repr.router,
|
|
|
- dns_servers: dns_servers,
|
|
|
+ dns_servers,
|
|
|
+ packet: None,
|
|
|
};
|
|
|
|
|
|
// RFC 2131 indicates clients should renew a lease halfway through its expiration.
|
|
@@ -410,6 +476,7 @@ impl Socket {
|
|
|
let mut dhcp_repr = DhcpRepr {
|
|
|
message_type: DhcpMessageType::Discover,
|
|
|
transaction_id: next_transaction_id,
|
|
|
+ secs: 0,
|
|
|
client_hardware_address: ethernet_addr,
|
|
|
client_ip: Ipv4Address::UNSPECIFIED,
|
|
|
your_ip: Ipv4Address::UNSPECIFIED,
|
|
@@ -421,10 +488,14 @@ impl Socket {
|
|
|
requested_ip: None,
|
|
|
client_identifier: Some(ethernet_addr),
|
|
|
server_identifier: None,
|
|
|
- parameter_request_list: Some(PARAMETER_REQUEST_LIST),
|
|
|
+ parameter_request_list: Some(
|
|
|
+ self.parameter_request_list
|
|
|
+ .unwrap_or(DEFAULT_PARAMETER_REQUEST_LIST),
|
|
|
+ ),
|
|
|
max_size: Some((cx.ip_mtu() - MAX_IPV4_HEADER_LEN - UDP_HEADER_LEN) as u16),
|
|
|
lease_duration: None,
|
|
|
dns_servers: None,
|
|
|
+ additional_options: self.outgoing_options,
|
|
|
};
|
|
|
|
|
|
let udp_repr = UdpRepr {
|
|
@@ -456,7 +527,7 @@ impl Socket {
|
|
|
emit(cx, (ipv4_repr, udp_repr, dhcp_repr))?;
|
|
|
|
|
|
// Update state AFTER the packet has been successfully sent.
|
|
|
- state.retry_at = cx.now() + DISCOVER_TIMEOUT;
|
|
|
+ state.retry_at = cx.now() + self.retry_config.discover_timeout;
|
|
|
self.transaction_id = next_transaction_id;
|
|
|
Ok(())
|
|
|
}
|
|
@@ -465,7 +536,7 @@ impl Socket {
|
|
|
return Ok(());
|
|
|
}
|
|
|
|
|
|
- if state.retry >= REQUEST_RETRIES {
|
|
|
+ if state.retry >= self.retry_config.request_retries {
|
|
|
net_debug!("DHCP request retries exceeded, restarting discovery");
|
|
|
self.reset();
|
|
|
return Ok(());
|
|
@@ -484,7 +555,8 @@ impl Socket {
|
|
|
emit(cx, (ipv4_repr, udp_repr, dhcp_repr))?;
|
|
|
|
|
|
// Exponential backoff: Double every 2 retries.
|
|
|
- state.retry_at = cx.now() + (REQUEST_TIMEOUT << (state.retry as u32 / 2));
|
|
|
+ state.retry_at = cx.now()
|
|
|
+ + (self.retry_config.initial_request_timeout << (state.retry as u32 / 2));
|
|
|
state.retry += 1;
|
|
|
|
|
|
self.transaction_id = next_transaction_id;
|
|
@@ -503,7 +575,7 @@ impl Socket {
|
|
|
}
|
|
|
|
|
|
ipv4_repr.src_addr = state.config.address.address();
|
|
|
- ipv4_repr.dst_addr = state.server.address;
|
|
|
+ ipv4_repr.dst_addr = state.config.server.address;
|
|
|
dhcp_repr.message_type = DhcpMessageType::Request;
|
|
|
dhcp_repr.client_ip = state.config.address.address();
|
|
|
|
|
@@ -516,8 +588,11 @@ impl Socket {
|
|
|
// of the remaining time until T2 (in RENEWING state) and one-half of
|
|
|
// the remaining lease time (in REBINDING state), down to a minimum of
|
|
|
// 60 seconds, before retransmitting the DHCPREQUEST message.
|
|
|
- state.renew_at =
|
|
|
- cx.now() + MIN_RENEW_TIMEOUT.max((state.expires_at - cx.now()) / 2);
|
|
|
+ state.renew_at = cx.now()
|
|
|
+ + self
|
|
|
+ .retry_config
|
|
|
+ .min_renew_timeout
|
|
|
+ .max((state.expires_at - cx.now()) / 2);
|
|
|
|
|
|
self.transaction_id = next_transaction_id;
|
|
|
Ok(())
|
|
@@ -548,7 +623,16 @@ impl Socket {
|
|
|
None
|
|
|
} else if let ClientState::Renewing(state) = &self.state {
|
|
|
self.config_changed = false;
|
|
|
- Some(Event::Configured(state.config))
|
|
|
+ Some(Event::Configured(Config {
|
|
|
+ server: state.config.server,
|
|
|
+ address: state.config.address,
|
|
|
+ router: state.config.router,
|
|
|
+ dns_servers: state.config.dns_servers,
|
|
|
+ packet: self
|
|
|
+ .receive_packet_buffer
|
|
|
+ .as_deref()
|
|
|
+ .map(DhcpPacket::new_unchecked),
|
|
|
+ }))
|
|
|
} else {
|
|
|
self.config_changed = false;
|
|
|
Some(Event::Deconfigured)
|
|
@@ -594,12 +678,12 @@ mod test {
|
|
|
// Helper functions
|
|
|
|
|
|
struct TestSocket {
|
|
|
- socket: Socket,
|
|
|
+ socket: Socket<'static>,
|
|
|
cx: Context<'static>,
|
|
|
}
|
|
|
|
|
|
impl Deref for TestSocket {
|
|
|
- type Target = Socket;
|
|
|
+ type Target = Socket<'static>;
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
&self.socket
|
|
|
}
|
|
@@ -741,6 +825,7 @@ mod test {
|
|
|
const DHCP_DEFAULT: DhcpRepr = DhcpRepr {
|
|
|
message_type: DhcpMessageType::Unknown(99),
|
|
|
transaction_id: TXID,
|
|
|
+ secs: 0,
|
|
|
client_hardware_address: MY_MAC,
|
|
|
client_ip: Ipv4Address::UNSPECIFIED,
|
|
|
your_ip: Ipv4Address::UNSPECIFIED,
|
|
@@ -756,6 +841,7 @@ mod test {
|
|
|
dns_servers: None,
|
|
|
max_size: None,
|
|
|
lease_duration: None,
|
|
|
+ additional_options: None,
|
|
|
};
|
|
|
|
|
|
const DHCP_DISCOVER: DhcpRepr = DhcpRepr {
|
|
@@ -840,13 +926,14 @@ mod test {
|
|
|
let mut s = socket();
|
|
|
s.state = ClientState::Renewing(RenewState {
|
|
|
config: Config {
|
|
|
+ server: ServerInfo {
|
|
|
+ address: SERVER_IP,
|
|
|
+ identifier: SERVER_IP,
|
|
|
+ },
|
|
|
address: Ipv4Cidr::new(MY_IP, 24),
|
|
|
dns_servers: DNS_IPS,
|
|
|
router: Some(SERVER_IP),
|
|
|
- },
|
|
|
- server: ServerInfo {
|
|
|
- address: SERVER_IP,
|
|
|
- identifier: SERVER_IP,
|
|
|
+ packet: None,
|
|
|
},
|
|
|
renew_at: Instant::from_secs(500),
|
|
|
expires_at: Instant::from_secs(1000),
|
|
@@ -870,9 +957,14 @@ mod test {
|
|
|
assert_eq!(
|
|
|
s.poll(),
|
|
|
Some(Event::Configured(Config {
|
|
|
+ server: ServerInfo {
|
|
|
+ address: SERVER_IP,
|
|
|
+ identifier: SERVER_IP,
|
|
|
+ },
|
|
|
address: Ipv4Cidr::new(MY_IP, 24),
|
|
|
dns_servers: DNS_IPS,
|
|
|
router: Some(SERVER_IP),
|
|
|
+ packet: None,
|
|
|
}))
|
|
|
);
|
|
|
|