|
@@ -9,6 +9,7 @@ use crate::wire::{
|
|
|
UdpRepr, DHCP_CLIENT_PORT, DHCP_MAX_DNS_SERVER_COUNT, DHCP_SERVER_PORT, UDP_HEADER_LEN,
|
|
|
};
|
|
|
use crate::wire::{DhcpOption, HardwareAddress};
|
|
|
+use heapless::Vec;
|
|
|
|
|
|
#[cfg(feature = "async")]
|
|
|
use super::WakerRegistration;
|
|
@@ -24,7 +25,7 @@ const DEFAULT_PARAMETER_REQUEST_LIST: &[u8] = &[
|
|
|
];
|
|
|
|
|
|
/// IPv4 configuration data provided by the DHCP server.
|
|
|
-#[derive(Debug, Eq, PartialEq, Clone, Copy)]
|
|
|
+#[derive(Debug, Eq, PartialEq, Clone)]
|
|
|
#[cfg_attr(feature = "defmt", derive(defmt::Format))]
|
|
|
pub struct Config<'a> {
|
|
|
/// Information on how to reach the DHCP server that responded with DHCP
|
|
@@ -36,7 +37,7 @@ pub struct Config<'a> {
|
|
|
/// match the DHCP server's address.
|
|
|
pub router: Option<Ipv4Address>,
|
|
|
/// DNS servers
|
|
|
- pub dns_servers: [Option<Ipv4Address>; DHCP_MAX_DNS_SERVER_COUNT],
|
|
|
+ pub dns_servers: Vec<Ipv4Address, DHCP_MAX_DNS_SERVER_COUNT>,
|
|
|
/// Received DHCP packet
|
|
|
pub packet: Option<DhcpPacket<&'a [u8]>>,
|
|
|
}
|
|
@@ -417,17 +418,19 @@ impl<'a> Socket<'a> {
|
|
|
|
|
|
// Cleanup the DNS servers list, keeping only unicasts/
|
|
|
// TP-Link TD-W8970 sends 0.0.0.0 as second DNS server if there's only one configured :(
|
|
|
- let mut dns_servers = [None; DHCP_MAX_DNS_SERVER_COUNT];
|
|
|
- if let Some(received) = dhcp_repr.dns_servers {
|
|
|
- let mut i = 0;
|
|
|
- for addr in received.iter().flatten() {
|
|
|
- if addr.is_unicast() {
|
|
|
- // This can never be out-of-bounds since both arrays have length DHCP_MAX_DNS_SERVER_COUNT
|
|
|
- dns_servers[i] = Some(*addr);
|
|
|
- i += 1;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
+ let mut dns_servers = Vec::new();
|
|
|
+
|
|
|
+ dhcp_repr
|
|
|
+ .dns_servers
|
|
|
+ .iter()
|
|
|
+ .flatten()
|
|
|
+ .filter(|s| s.is_unicast())
|
|
|
+ .for_each(|a| {
|
|
|
+ // This will never produce an error, as both the arrays and `dns_servers`
|
|
|
+ // have length DHCP_MAX_DNS_SERVER_COUNT
|
|
|
+ dns_servers.push(*a).ok();
|
|
|
+ });
|
|
|
+
|
|
|
let config = Config {
|
|
|
server,
|
|
|
address: Ipv4Cidr::new(dhcp_repr.your_ip, prefix_len),
|
|
@@ -627,7 +630,7 @@ impl<'a> Socket<'a> {
|
|
|
server: state.config.server,
|
|
|
address: state.config.address,
|
|
|
router: state.config.router,
|
|
|
- dns_servers: state.config.dns_servers,
|
|
|
+ dns_servers: state.config.dns_servers.clone(),
|
|
|
packet: self
|
|
|
.receive_packet_buffer
|
|
|
.as_deref()
|
|
@@ -775,8 +778,8 @@ mod test {
|
|
|
const DNS_IP_1: Ipv4Address = Ipv4Address([1, 1, 1, 1]);
|
|
|
const DNS_IP_2: Ipv4Address = Ipv4Address([1, 1, 1, 2]);
|
|
|
const DNS_IP_3: Ipv4Address = Ipv4Address([1, 1, 1, 3]);
|
|
|
- const DNS_IPS: [Option<Ipv4Address>; DHCP_MAX_DNS_SERVER_COUNT] =
|
|
|
- [Some(DNS_IP_1), Some(DNS_IP_2), Some(DNS_IP_3)];
|
|
|
+ const DNS_IPS: &[Ipv4Address] = &[DNS_IP_1, DNS_IP_2, DNS_IP_3];
|
|
|
+
|
|
|
const MASK_24: Ipv4Address = Ipv4Address([255, 255, 255, 0]);
|
|
|
|
|
|
const MY_MAC: EthernetAddress = EthernetAddress([0x02, 0x02, 0x02, 0x02, 0x02, 0x02]);
|
|
@@ -852,19 +855,21 @@ mod test {
|
|
|
..DHCP_DEFAULT
|
|
|
};
|
|
|
|
|
|
- const DHCP_OFFER: DhcpRepr = DhcpRepr {
|
|
|
- message_type: DhcpMessageType::Offer,
|
|
|
- server_ip: SERVER_IP,
|
|
|
- server_identifier: Some(SERVER_IP),
|
|
|
+ fn dhcp_offer() -> DhcpRepr<'static> {
|
|
|
+ DhcpRepr {
|
|
|
+ message_type: DhcpMessageType::Offer,
|
|
|
+ server_ip: SERVER_IP,
|
|
|
+ server_identifier: Some(SERVER_IP),
|
|
|
|
|
|
- your_ip: MY_IP,
|
|
|
- router: Some(SERVER_IP),
|
|
|
- subnet_mask: Some(MASK_24),
|
|
|
- dns_servers: Some(DNS_IPS),
|
|
|
- lease_duration: Some(1000),
|
|
|
+ your_ip: MY_IP,
|
|
|
+ router: Some(SERVER_IP),
|
|
|
+ subnet_mask: Some(MASK_24),
|
|
|
+ dns_servers: Some(Vec::from_slice(DNS_IPS).unwrap()),
|
|
|
+ lease_duration: Some(1000),
|
|
|
|
|
|
- ..DHCP_DEFAULT
|
|
|
- };
|
|
|
+ ..DHCP_DEFAULT
|
|
|
+ }
|
|
|
+ }
|
|
|
|
|
|
const DHCP_REQUEST: DhcpRepr = DhcpRepr {
|
|
|
message_type: DhcpMessageType::Request,
|
|
@@ -877,19 +882,21 @@ mod test {
|
|
|
..DHCP_DEFAULT
|
|
|
};
|
|
|
|
|
|
- const DHCP_ACK: DhcpRepr = DhcpRepr {
|
|
|
- message_type: DhcpMessageType::Ack,
|
|
|
- server_ip: SERVER_IP,
|
|
|
- server_identifier: Some(SERVER_IP),
|
|
|
+ fn dhcp_ack() -> DhcpRepr<'static> {
|
|
|
+ DhcpRepr {
|
|
|
+ message_type: DhcpMessageType::Ack,
|
|
|
+ server_ip: SERVER_IP,
|
|
|
+ server_identifier: Some(SERVER_IP),
|
|
|
|
|
|
- your_ip: MY_IP,
|
|
|
- router: Some(SERVER_IP),
|
|
|
- subnet_mask: Some(MASK_24),
|
|
|
- dns_servers: Some(DNS_IPS),
|
|
|
- lease_duration: Some(1000),
|
|
|
+ your_ip: MY_IP,
|
|
|
+ router: Some(SERVER_IP),
|
|
|
+ subnet_mask: Some(MASK_24),
|
|
|
+ dns_servers: Some(Vec::from_slice(DNS_IPS).unwrap()),
|
|
|
+ lease_duration: Some(1000),
|
|
|
|
|
|
- ..DHCP_DEFAULT
|
|
|
- };
|
|
|
+ ..DHCP_DEFAULT
|
|
|
+ }
|
|
|
+ }
|
|
|
|
|
|
const DHCP_NAK: DhcpRepr = DhcpRepr {
|
|
|
message_type: DhcpMessageType::Nak,
|
|
@@ -931,7 +938,7 @@ mod test {
|
|
|
identifier: SERVER_IP,
|
|
|
},
|
|
|
address: Ipv4Cidr::new(MY_IP, 24),
|
|
|
- dns_servers: DNS_IPS,
|
|
|
+ dns_servers: Vec::from_slice(DNS_IPS).unwrap(),
|
|
|
router: Some(SERVER_IP),
|
|
|
packet: None,
|
|
|
},
|
|
@@ -948,11 +955,11 @@ mod test {
|
|
|
|
|
|
recv!(s, [(IP_BROADCAST, UDP_SEND, DHCP_DISCOVER)]);
|
|
|
assert_eq!(s.poll(), None);
|
|
|
- send!(s, (IP_RECV, UDP_RECV, DHCP_OFFER));
|
|
|
+ send!(s, (IP_RECV, UDP_RECV, dhcp_offer()));
|
|
|
assert_eq!(s.poll(), None);
|
|
|
recv!(s, [(IP_BROADCAST, UDP_SEND, DHCP_REQUEST)]);
|
|
|
assert_eq!(s.poll(), None);
|
|
|
- send!(s, (IP_RECV, UDP_RECV, DHCP_ACK));
|
|
|
+ send!(s, (IP_RECV, UDP_RECV, dhcp_ack()));
|
|
|
|
|
|
assert_eq!(
|
|
|
s.poll(),
|
|
@@ -962,7 +969,7 @@ mod test {
|
|
|
identifier: SERVER_IP,
|
|
|
},
|
|
|
address: Ipv4Cidr::new(MY_IP, 24),
|
|
|
- dns_servers: DNS_IPS,
|
|
|
+ dns_servers: Vec::from_slice(DNS_IPS).unwrap(),
|
|
|
router: Some(SERVER_IP),
|
|
|
packet: None,
|
|
|
}))
|
|
@@ -988,7 +995,7 @@ mod test {
|
|
|
recv!(s, time 20_000, [(IP_BROADCAST, UDP_SEND, DHCP_DISCOVER)]);
|
|
|
|
|
|
// check after retransmits it still works
|
|
|
- send!(s, time 20_000, (IP_RECV, UDP_RECV, DHCP_OFFER));
|
|
|
+ send!(s, time 20_000, (IP_RECV, UDP_RECV, dhcp_offer()));
|
|
|
recv!(s, time 20_000, [(IP_BROADCAST, UDP_SEND, DHCP_REQUEST)]);
|
|
|
}
|
|
|
|
|
@@ -997,7 +1004,7 @@ mod test {
|
|
|
let mut s = socket();
|
|
|
|
|
|
recv!(s, time 0, [(IP_BROADCAST, UDP_SEND, DHCP_DISCOVER)]);
|
|
|
- send!(s, time 0, (IP_RECV, UDP_RECV, DHCP_OFFER));
|
|
|
+ send!(s, time 0, (IP_RECV, UDP_RECV, dhcp_offer()));
|
|
|
recv!(s, time 0, [(IP_BROADCAST, UDP_SEND, DHCP_REQUEST)]);
|
|
|
recv!(s, time 1_000, []);
|
|
|
recv!(s, time 5_000, [(IP_BROADCAST, UDP_SEND, DHCP_REQUEST)]);
|
|
@@ -1007,7 +1014,7 @@ mod test {
|
|
|
recv!(s, time 20_000, [(IP_BROADCAST, UDP_SEND, DHCP_REQUEST)]);
|
|
|
|
|
|
// check after retransmits it still works
|
|
|
- send!(s, time 20_000, (IP_RECV, UDP_RECV, DHCP_ACK));
|
|
|
+ send!(s, time 20_000, (IP_RECV, UDP_RECV, dhcp_ack()));
|
|
|
|
|
|
match &s.state {
|
|
|
ClientState::Renewing(r) => {
|
|
@@ -1023,7 +1030,7 @@ mod test {
|
|
|
let mut s = socket();
|
|
|
|
|
|
recv!(s, time 0, [(IP_BROADCAST, UDP_SEND, DHCP_DISCOVER)]);
|
|
|
- send!(s, time 0, (IP_RECV, UDP_RECV, DHCP_OFFER));
|
|
|
+ send!(s, time 0, (IP_RECV, UDP_RECV, dhcp_offer()));
|
|
|
recv!(s, time 0, [(IP_BROADCAST, UDP_SEND, DHCP_REQUEST)]);
|
|
|
recv!(s, time 5_000, [(IP_BROADCAST, UDP_SEND, DHCP_REQUEST)]);
|
|
|
recv!(s, time 10_000, [(IP_BROADCAST, UDP_SEND, DHCP_REQUEST)]);
|
|
@@ -1035,7 +1042,7 @@ mod test {
|
|
|
recv!(s, time 70_000, [(IP_BROADCAST, UDP_SEND, DHCP_DISCOVER)]);
|
|
|
|
|
|
// check it still works
|
|
|
- send!(s, time 60_000, (IP_RECV, UDP_RECV, DHCP_OFFER));
|
|
|
+ send!(s, time 60_000, (IP_RECV, UDP_RECV, dhcp_offer()));
|
|
|
recv!(s, time 60_000, [(IP_BROADCAST, UDP_SEND, DHCP_REQUEST)]);
|
|
|
}
|
|
|
|
|
@@ -1044,7 +1051,7 @@ mod test {
|
|
|
let mut s = socket();
|
|
|
|
|
|
recv!(s, time 0, [(IP_BROADCAST, UDP_SEND, DHCP_DISCOVER)]);
|
|
|
- send!(s, time 0, (IP_RECV, UDP_RECV, DHCP_OFFER));
|
|
|
+ send!(s, time 0, (IP_RECV, UDP_RECV, dhcp_offer()));
|
|
|
recv!(s, time 0, [(IP_BROADCAST, UDP_SEND, DHCP_REQUEST)]);
|
|
|
send!(s, time 0, (IP_SERVER_BROADCAST, UDP_RECV, DHCP_NAK));
|
|
|
recv!(s, time 0, [(IP_BROADCAST, UDP_SEND, DHCP_DISCOVER)]);
|
|
@@ -1068,7 +1075,7 @@ mod test {
|
|
|
_ => panic!("Invalid state"),
|
|
|
}
|
|
|
|
|
|
- send!(s, time 500_000, (IP_RECV, UDP_RECV, DHCP_ACK));
|
|
|
+ send!(s, time 500_000, (IP_RECV, UDP_RECV, dhcp_ack()));
|
|
|
assert_eq!(s.poll(), None);
|
|
|
|
|
|
match &s.state {
|
|
@@ -1093,7 +1100,7 @@ mod test {
|
|
|
recv!(s, time 875_000, [(IP_SEND, UDP_SEND, DHCP_RENEW)]);
|
|
|
|
|
|
// check it still works
|
|
|
- send!(s, time 875_000, (IP_RECV, UDP_RECV, DHCP_ACK));
|
|
|
+ send!(s, time 875_000, (IP_RECV, UDP_RECV, dhcp_ack()));
|
|
|
match &s.state {
|
|
|
ClientState::Renewing(r) => {
|
|
|
// NOW the expiration gets bumped
|