Browse Source

Listening TCP sockets should reject, not drop, ACKs.

whitequark 7 years ago
parent
commit
56ee8e5df8
1 changed files with 13 additions and 10 deletions
  1. 13 10
      src/socket/tcp.rs

+ 13 - 10
src/socket/tcp.rs

@@ -690,6 +690,11 @@ impl<'a> TcpSocket<'a> {
         let packet = TcpPacket::new_checked(&payload[..ip_repr.payload_len()])?;
         let packet = TcpPacket::new_checked(&payload[..ip_repr.payload_len()])?;
         let repr = TcpRepr::parse(&packet, &ip_repr.src_addr(), &ip_repr.dst_addr())?;
         let repr = TcpRepr::parse(&packet, &ip_repr.src_addr(), &ip_repr.dst_addr())?;
 
 
+        // If we're still listening for SYNs and the packet has an ACK, it cannot
+        // be destined to this socket, but another one may well listen on the same
+        // local endpoint.
+        if self.state == State::Listen && repr.ack_number.is_some() { return Err(Error::Rejected) }
+
         // Reject packets with a wrong destination.
         // Reject packets with a wrong destination.
         if self.local_endpoint.port != repr.dst_port { return Err(Error::Rejected) }
         if self.local_endpoint.port != repr.dst_port { return Err(Error::Rejected) }
         if !self.local_endpoint.addr.is_unspecified() &&
         if !self.local_endpoint.addr.is_unspecified() &&
@@ -715,17 +720,11 @@ impl<'a> TcpSocket<'a> {
 
 
         // Reject unacceptable acknowledgements.
         // Reject unacceptable acknowledgements.
         match (self.state, repr) {
         match (self.state, repr) {
-            // The initial SYN (or whatever) cannot contain an acknowledgement.
-            // It may be destined to another socket though.
-            (State::Listen, TcpRepr { ack_number: Some(_), .. }) => {
-                net_debug!("[{}]{}:{}: unacceptable ACK in LISTEN state",
-                           self.debug_id, self.local_endpoint, self.remote_endpoint);
-                return Err(Error::Dropped)
-            }
-            (State::Listen, TcpRepr { ack_number: None, .. }) => (),
             // An RST received in response to initial SYN is acceptable if it acknowledges
             // An RST received in response to initial SYN is acceptable if it acknowledges
             // the initial SYN.
             // the initial SYN.
-            (State::SynSent, TcpRepr { control: TcpControl::Rst, ack_number: None, .. }) => {
+            (State::SynSent, TcpRepr {
+                control: TcpControl::Rst, ack_number: None, ..
+            }) => {
                 net_debug!("[{}]{}:{}: unacceptable RST (expecting RST|ACK) \
                 net_debug!("[{}]{}:{}: unacceptable RST (expecting RST|ACK) \
                             in response to initial SYN",
                             in response to initial SYN",
                            self.debug_id, self.local_endpoint, self.remote_endpoint);
                            self.debug_id, self.local_endpoint, self.remote_endpoint);
@@ -742,6 +741,10 @@ impl<'a> TcpSocket<'a> {
             }
             }
             // Any other RST need only have a valid sequence number.
             // Any other RST need only have a valid sequence number.
             (_, TcpRepr { control: TcpControl::Rst, .. }) => (),
             (_, TcpRepr { control: TcpControl::Rst, .. }) => (),
+            // The initial SYN cannot contain an acknowledgement.
+            (State::Listen, TcpRepr { ack_number: None, .. }) => (),
+            // This case is handled above.
+            (State::Listen, TcpRepr { ack_number: Some(_), .. }) => unreachable!(),
             // Every packet after the initial SYN must be an acknowledgement.
             // Every packet after the initial SYN must be an acknowledgement.
             (_, TcpRepr { ack_number: None, .. }) => {
             (_, TcpRepr { ack_number: None, .. }) => {
                 net_debug!("[{}]{}:{}: expecting an ACK",
                 net_debug!("[{}]{}:{}: expecting an ACK",
@@ -1522,7 +1525,7 @@ mod test {
             seq_number: REMOTE_SEQ,
             seq_number: REMOTE_SEQ,
             ack_number: Some(LOCAL_SEQ),
             ack_number: Some(LOCAL_SEQ),
             ..SEND_TEMPL
             ..SEND_TEMPL
-        }, Err(Error::Dropped));
+        }, Err(Error::Rejected));
         assert_eq!(s.state, State::Listen);
         assert_eq!(s.state, State::Listen);
     }
     }