|
@@ -179,6 +179,8 @@ pub struct TcpSocket<'a> {
|
|
|
timeout: Option<u64>,
|
|
|
/// Interval at which keep-alive packets will be sent.
|
|
|
keep_alive: Option<u64>,
|
|
|
+ /// The time-to-live (IPv4) or hop limit (IPv6) value used in outgoing packets.
|
|
|
+ ttl: Option<u8>,
|
|
|
/// Address passed to listen(). Listen address is set when listen() is called and
|
|
|
/// used every time the socket is reset back to the LISTEN state.
|
|
|
listen_address: IpAddress,
|
|
@@ -236,6 +238,7 @@ impl<'a> TcpSocket<'a> {
|
|
|
rx_buffer: rx_buffer,
|
|
|
timeout: None,
|
|
|
keep_alive: None,
|
|
|
+ ttl: None,
|
|
|
listen_address: IpAddress::default(),
|
|
|
local_endpoint: IpEndpoint::default(),
|
|
|
remote_endpoint: IpEndpoint::default(),
|
|
@@ -311,6 +314,33 @@ impl<'a> TcpSocket<'a> {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+ /// Return the time-to-live (IPv4) or hop limit (IPv6) value used in outgoing packets.
|
|
|
+ ///
|
|
|
+ /// See also the [set_ttl](#method.set_ttl) method
|
|
|
+ pub fn ttl(&self) -> Option<u8> {
|
|
|
+ self.ttl
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Set the time-to-live (IPv4) or hop limit (IPv6) value used in outgoing packets.
|
|
|
+ ///
|
|
|
+ /// A socket without an explicitly set TTL value uses the default [IANA recommended]
|
|
|
+ /// value (`64`).
|
|
|
+ ///
|
|
|
+ /// # Panics
|
|
|
+ ///
|
|
|
+ /// This function panics if a TTL value of `0` is given. See [RFC 1122 § 3.2.1.7].
|
|
|
+ ///
|
|
|
+ /// [IANA recommended]: https://www.iana.org/assignments/ip-parameters/ip-parameters.xhtml
|
|
|
+ /// [RFC 1122 § 3.2.1.7]: https://tools.ietf.org/html/rfc1122#section-3.2.1.7
|
|
|
+ pub fn set_ttl(&mut self, ttl: Option<u8>) {
|
|
|
+ // A host MUST NOT send a datagram with a Time-to-Live (TTL)
|
|
|
+ // value of 0
|
|
|
+ match ttl {
|
|
|
+ Some(0) => { panic!("A TTL value of 0 is invalid for a sent packet"); },
|
|
|
+ catchall => self.ttl = catchall,
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
/// Return the local endpoint.
|
|
|
#[inline]
|
|
|
pub fn local_endpoint(&self) -> IpEndpoint {
|
|
@@ -337,6 +367,7 @@ impl<'a> TcpSocket<'a> {
|
|
|
self.rx_buffer.clear();
|
|
|
self.keep_alive = None;
|
|
|
self.timeout = None;
|
|
|
+ self.ttl = None;
|
|
|
self.listen_address = IpAddress::default();
|
|
|
self.local_endpoint = IpEndpoint::default();
|
|
|
self.remote_endpoint = IpEndpoint::default();
|
|
@@ -733,7 +764,8 @@ impl<'a> TcpSocket<'a> {
|
|
|
src_addr: ip_repr.dst_addr(),
|
|
|
dst_addr: ip_repr.src_addr(),
|
|
|
protocol: IpProtocol::Tcp,
|
|
|
- payload_len: reply_repr.buffer_len()
|
|
|
+ payload_len: reply_repr.buffer_len(),
|
|
|
+ ttl: 64
|
|
|
};
|
|
|
(ip_reply_repr, reply_repr)
|
|
|
}
|
|
@@ -1239,6 +1271,7 @@ impl<'a> TcpSocket<'a> {
|
|
|
src_addr: self.local_endpoint.addr,
|
|
|
dst_addr: self.remote_endpoint.addr,
|
|
|
protocol: IpProtocol::Tcp,
|
|
|
+ ttl: self.ttl.unwrap_or(64),
|
|
|
payload_len: 0
|
|
|
}.lower(&[])?;
|
|
|
|
|
@@ -1447,7 +1480,8 @@ impl<'a> fmt::Write for TcpSocket<'a> {
|
|
|
|
|
|
#[cfg(test)]
|
|
|
mod test {
|
|
|
- use wire::{IpAddress, Ipv4Address, IpCidr};
|
|
|
+ use wire::{IpAddress, IpRepr};
|
|
|
+ use wire::{Ipv4Address, IpCidr, Ipv4Repr};
|
|
|
use super::*;
|
|
|
|
|
|
#[test]
|
|
@@ -1479,7 +1513,8 @@ mod test {
|
|
|
|
|
|
const SEND_IP_TEMPL: IpRepr = IpRepr::Unspecified {
|
|
|
src_addr: LOCAL_IP, dst_addr: REMOTE_IP,
|
|
|
- protocol: IpProtocol::Tcp, payload_len: 20
|
|
|
+ protocol: IpProtocol::Tcp, payload_len: 20,
|
|
|
+ ttl: 64
|
|
|
};
|
|
|
const SEND_TEMPL: TcpRepr<'static> = TcpRepr {
|
|
|
src_port: REMOTE_PORT, dst_port: LOCAL_PORT,
|
|
@@ -1490,7 +1525,8 @@ mod test {
|
|
|
};
|
|
|
const _RECV_IP_TEMPL: IpRepr = IpRepr::Unspecified {
|
|
|
src_addr: REMOTE_IP, dst_addr: LOCAL_IP,
|
|
|
- protocol: IpProtocol::Tcp, payload_len: 20
|
|
|
+ protocol: IpProtocol::Tcp, payload_len: 20,
|
|
|
+ ttl: 64
|
|
|
};
|
|
|
const RECV_TEMPL: TcpRepr<'static> = TcpRepr {
|
|
|
src_port: LOCAL_PORT, dst_port: REMOTE_PORT,
|
|
@@ -1506,7 +1542,8 @@ mod test {
|
|
|
src_addr: REMOTE_IP,
|
|
|
dst_addr: LOCAL_IP,
|
|
|
protocol: IpProtocol::Tcp,
|
|
|
- payload_len: repr.buffer_len()
|
|
|
+ payload_len: repr.buffer_len(),
|
|
|
+ ttl: 64
|
|
|
};
|
|
|
trace!("send: {}", repr);
|
|
|
|
|
@@ -3470,7 +3507,8 @@ mod test {
|
|
|
src_addr: REMOTE_IP,
|
|
|
dst_addr: LOCAL_IP,
|
|
|
protocol: IpProtocol::Tcp,
|
|
|
- payload_len: tcp_repr.buffer_len()
|
|
|
+ payload_len: tcp_repr.buffer_len(),
|
|
|
+ ttl: 64
|
|
|
};
|
|
|
assert!(s.accepts(&ip_repr, &tcp_repr));
|
|
|
|
|
@@ -3478,7 +3516,8 @@ mod test {
|
|
|
src_addr: OTHER_IP,
|
|
|
dst_addr: LOCAL_IP,
|
|
|
protocol: IpProtocol::Tcp,
|
|
|
- payload_len: tcp_repr.buffer_len()
|
|
|
+ payload_len: tcp_repr.buffer_len(),
|
|
|
+ ttl: 64
|
|
|
};
|
|
|
assert!(!s.accepts(&ip_repr_wrong_src, &tcp_repr));
|
|
|
|
|
@@ -3486,8 +3525,28 @@ mod test {
|
|
|
src_addr: REMOTE_IP,
|
|
|
dst_addr: OTHER_IP,
|
|
|
protocol: IpProtocol::Tcp,
|
|
|
- payload_len: tcp_repr.buffer_len()
|
|
|
+ payload_len: tcp_repr.buffer_len(),
|
|
|
+ ttl: 64
|
|
|
};
|
|
|
assert!(!s.accepts(&ip_repr_wrong_dst, &tcp_repr));
|
|
|
}
|
|
|
+
|
|
|
+ #[test]
|
|
|
+ fn test_set_ttl() {
|
|
|
+ let mut s = socket_syn_received();
|
|
|
+ let mut caps = DeviceCapabilities::default();
|
|
|
+ caps.max_transmission_unit = 1520;
|
|
|
+
|
|
|
+ s.set_ttl(Some(0x2a));
|
|
|
+ assert_eq!(s.dispatch(0, &caps, |(ip_repr, _)| {
|
|
|
+ assert_eq!(ip_repr, IpRepr::Ipv4(Ipv4Repr {
|
|
|
+ src_addr: Ipv4Address([10, 0, 0, 1]),
|
|
|
+ dst_addr: Ipv4Address([10, 0, 0, 2]),
|
|
|
+ protocol: IpProtocol::Tcp,
|
|
|
+ payload_len: 24,
|
|
|
+ ttl: 0x2a,
|
|
|
+ }));
|
|
|
+ Ok(())
|
|
|
+ }), Ok(()));
|
|
|
+ }
|
|
|
}
|