Skip site navigation (1)Skip section navigation (2)
Date:      Sat, 31 May 2008 20:02:35 GMT
From:      Andre Oppermann <andre@FreeBSD.org>
To:        Perforce Change Reviews <perforce@freebsd.org>
Subject:   PERFORCE change 142643 for review
Message-ID:  <200805312002.m4VK2ZrH067296@repoman.freebsd.org>

next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=142643

Change 142643 by andre@andre_flirtbox on 2008/05/31 20:01:48

	WIP Checkpoint of tcp_do_segment() reworking.
	
	o Some further major rototiling.
	o Get data flow structure firmly in place.
	o Add extensive comments.
	o Add extensive RFC references to almost any part.
	o Validate a large part against relevant RFCs.
	o Remaining work and unclarities marked with XXXAO.
	o Does not yet compile.
	o Window update and timestamp code not yet complete.

Affected files ...

.. //depot/projects/tcp_new/netinet/tcp_input.c#5 edit

Differences ...

==== //depot/projects/tcp_new/netinet/tcp_input.c#5 (text+ko) ====

@@ -154,7 +154,10 @@
 static void	 tcp_do_segment(struct mbuf *, struct tcphdr *,
 		     struct socket *, struct tcpcb *, int, int);
 static void	 tcp_do_time(struct tcpcb *tp, struct tcphdr *th,
-		     struct tcpopt *to);
+		     struct tcpopt *to, int acked, int tlen);
+static void	 tcp_do_urg(struct tcpcb *tp, struct tcphdr *th, int tlen);
+static void	 tcp_do_wu(struct tcpcb *tp, struct tcphdr *th,
+		     struct tcpopt *to, int tiwin, int tlen);
 static void	 tcp_dropwithreset(struct mbuf *, struct tcphdr *,
 		     struct tcpcb *, int, int);
 static void	 tcp_pulloutofband(struct socket *,
@@ -881,24 +884,30 @@
 	INP_INFO_WLOCK_ASSERT(&tcbinfo);
 	INP_LOCK_ASSERT(tp->t_inpcb);
 	KASSERT(tp->t_state > TCPS_LISTEN,
-	    ("%s: TCPS_LISTEN", __func__));
+	    ("%s: TCPS_LISTEN invalid", __func__));
 	KASSERT(tp->t_state != TCPS_TIME_WAIT,
-	    ("%s: TCPS_TIME_WAIT", __func__));
+	    ("%s: TCPS_TIME_WAIT invalid", __func__));
 
 	/*
-	 * Store the flags in a variable for easy manipulation.
+	 * Store the flags in a variable for easy manipulation
+	 * and because we won't have access to th->th_flags in
+	 * later stages.
 	 */
 	thflags = th->th_flags;
 
 	/*
 	 * Unscale the window into a 32-bit value.
+	 *  RFC1122: section 4.2.2.3
+	 *  RFC1323bis: section 2.3
 	 * 
-	 * NB: For the SYN_SENT state the scale is zero.
+	 * NB: In SYN_SENT state the scale is zero.
 	 */
 	tiwin = th->th_win << tp->snd_scale;
 
 	/*
 	 * Parse options on any incoming segment (if present).
+	 *  RFC793: section 3.1, page 17-19
+	 *  RFC1122: section 4.2.2.5
 	 */
 	if ((th->th_off << 2) != sizeof(struct tcphdr))
 		tcp_dooptions(&to, (u_char *)(th + 1),
@@ -908,29 +917,31 @@
 		to.to_flags = 0;
 
 	/*
-	 * Normalize timestamp if syncookies were used when this
+	 * Normalize our timestamp if syncookies were used when this
 	 * connection was established.
 	 */
 	if (to.to_flags & TOF_TS)
 		to.to_tsecr -= tp->ts_offset;
 
 	/*
-	 * Calculate amount of space in receive window.
-	 * Receive window is amount of space in rcv queue,
-	 * but not less than advertised window.
+	 * Calculate amount of space in our receive window.
+	 * Receive window is the amount of space in rcv queue,
+	 * but not less than last advertised window.
+	 *  RFC793: section 3.7, page 42-44
+	 *  RFC1122: section 4.2.2.16
 	 */
 	rwin = sbspace(&so->so_rcv);
-	rwin = imax(rwin, (int)(tp->rcv_adv - tp->rcv_nxt));
+	rwin = imax(rwin, (int)(tp->rcv_advwin - tp->rcv_nxt));
 
 	/*
-	 * Validation checks.  We may get any shit here.  Have to be careful.
+	 * Validation checks on any incoming segment.
+	 * We may get anything here.  Have to be careful.
 	 */
 	switch (tp->t_state) {
 	/*
-	 * If the state is SYN_RECEIVED:
-	 *	syncache handled all validation, socket, inpcb and tcpcb
-	 *	setup for us.  All that is left is the state transition
-	 *	into established state and initializations of the timers.
+	 * Syncache handled all validation, socket, inpcb and tcpcb
+	 * setup for us.  All that is left is the state transition
+	 * into established state and initializations of the timers.
 	 */
 	case TCPS_SYN_RECEIVED:
 		tp->t_starttime = ticks;
@@ -943,74 +954,104 @@
 		break;
 
 	/*
-	 * If the state is SYN_SENT:
-	 *	if seg contains a RST, then drop the connection.
-	 *	if seg does not contain SYN and ACK, then drop it.
-	 *	if seg contains an ACK, but not for our SYN, drop the input.
-	 * Otherwise this is an acceptable SYN segment
-	 *	initialize tp->rcv_nxt and tp->irs
-	 *	if seg contains ack then advance tp->snd_una
-	 *	if SYN has been acked change to ESTABLISHED else SYN_RCVD state
-	 *	arrange for segment to be acked (eventually)
-	 *	continue processing rest of data/controls, beginning with URG
+	 * Validate the returned SYN-ACK, process the negotiated
+	 * options, complete the initialization of the tcpcb and
+	 * transition into ESTABLISHED state.
 	 */
 	case TCPS_SYN_SENT:
 		/*
-		 * RST is handled below.
+		 * RST is handled separately below.
+		 *  RFC793: section 3.9, page 66-67, second check
 		 */
 		if (thflags & TH_RST)
 			break;
 
 		/*
-		 * SYN|ACK must be present.
+		 * FIN is not valid yet and the segment to be ignored.
+		 *  RFC793: section 3.9, page 75, eighth check, first paragraph
+		 */
+		if (thflags & TH_FIN) {
+			tcplog("FIN invalid, segment ignored");
+			goto drop;
+		}
+
+		/*
+		 * SYN must be present.
+		 *  RFC793: section 3.9, page 67-68, fourth and fifth check
+		 *
+		 * NB: We have to remove SYN from thflags to
+		 * prevent the later bogus SYN check from
+		 * triggering.
 		 */
-		if (thflags & (TH_SYN|TH_ACK) != (TH_SYN|TH_ACK)) {
-			tcplog("Missing SYN|ACK, segment ignored");
+		if (!(thflags & TH_SYN)) {
+			tcplog("Missing SYN, segment ignored");
 			goto drop;
 		}
+		thflags &= ~TH_SYN;	/* SYN is processed. */
+
+		/*
+		 * ACK must be present.
+		 *  RFC793: section 3.9, page 67-68, fourth check
+		 *
+		 * XXXAO: Simultaneous open to be handled.
+		 *  RFC1122: section 4.2.2.10
+		 */
+		if (!(thflags & TH_ACK)) {
+			tcplog("Missing ACK, segment rejected");
+			goto dropwithreset;
+		}
 
 		/*
-		 * ACK must ack our ISN and any data we may
-		 * have sent with our SYN.
+		 * ACK must ack our ISN, SYN and possibly data we may
+		 * have sent with our SYN.  The latter is not strictly
+		 * necessary as many implementations chose to ignore
+		 * data in a SYN.  We want to make use of it again in
+		 * a future incarnation of T/TCP with a 'quick' 3WHS.
+		 *  RFC793: section 3.9, page 66, first check
 		 */
-		if (SEQ_LEQ(th->th_ack, tp->snd_iss) ||
-		    SEQ_GEQ(th->th_ack, tp->snd_nxt) ||
-		    SEQ_LT(th->th_ack, tp->snd_una)) {
-			tcplog("Incorrect ACK, segment rejected");
-			/* XXXAO: Close connection? Or ignore. */
+		if (SEQ_LEQ(th->th_ack, tp->snd_una) ||
+		    SEQ_GT(th->th_ack, tp->snd_nxt)) {
+			tcplog("Incorrect ACK, segment ignored");
 			goto dropwithreset;
 		}
 
 		/*
 		 * Option processing:
 		 *
-		 * If there wasn't a MSS option fall back to
-		 * default mss.
+		 * If there wasn't a MSS option fall back to default mss.
+		 * tcp_mss will calculate the largest possible MSS for us.
+		 * The MSS option is not a fully negotiated option and may
+		 * be returned even if we haven't sent it with our initial
+		 * SYN.  It is not common practice to do so however.
+		 *  RFC793: section 3.1, page 18-19
+		 *  RFC1122: section 4.2.2.6
+		 *  RFC1191: section 3.1
 		 */
-		if (!(tp->t_flags & TF_NOOPT) && (to.to_flags & TOF_MSS))
+		if (to.to_flags & TOF_MSS)
 			tcp_mss(tp, to.to_mss);
-		else if (tcp_do_path_mtu_discovery)
-			/* MTU of interface... */
 		else
 			tcp_mss(tp, tcp_mssdflt);
 
 		/*
 		 * Do window scaling on this connection?
+		 *  RFC1323bis: section 2.2 and 2.3
 		 *
-		 * NB: According to RFC1323 the window field
-		 * in a SYN (i.e., a <SYN> or <SYN,ACK>)
-		 * segment itself is never scaled.
+		 * NB: The window field in a SYN (i.e., a <SYN>
+		 * or <SYN,ACK>) segment itself is never scaled.
 		 */
 		if ((tp->t_flags & TF_WINSCALE) &&
 		    (to.to_flags & TOF_SCALE)) {
 			tp->snd_scale = to.to_wscale;
 		} else if (tp->t_flags & TF_WINSCALE) {
+			/* No window scaling. */
 			tp->t_flags &= ~TF_WINSCALE;
+			tp->snd_scale = 0;
 			tp->rcv_scale = 0;
 		} else if (to.to_flags & TOF_SCALE) {
 			/*
 			 * The remote end doesn't play right with us
 			 * and introduces options we haven't sent.
+			 *  RFC1323bis: section 1.3, second paragraph
 			 */
 			tcplog("Window Scaling Option unexpected, "
 			    "connection aborted");
@@ -1022,6 +1063,7 @@
 
 		/*
 		 * Do timestamps on this connection?
+		 *  RFC1323bis: section 3.2, first and last sentence
 		 */
 		if ((tp->t_flags & TF_TIMESTAMP) &&
 		    !(to.to_flags & TOF_TS))
@@ -1031,8 +1073,9 @@
 			/*
 			 * The remote end doesn't play right with us
 			 * and introduces options we haven't sent.
+			 *  RFC1323bis: section 1.3, second paragraph
 			 */
-			tcplog("Timestamp unexpected, "
+			tcplog("Timestamp Option unexpected, "
 			    "connection aborted");
 			tp->t_error = ENETRESET;
 			tp = tcp_close(tp);
@@ -1042,6 +1085,7 @@
 
 		/*
 		 * Do SACK on this connection?
+		 *  RFC2018: section 2
 		 */
 		if ((tp->t_flags & TF_SACK_PERMIT) &&
 		    !(to.to_flags & TOF_SACKPERM))
@@ -1051,6 +1095,7 @@
 			/*
 			 * The remote end doesn't play right with us
 			 * and introduces options we haven't sent.
+			 *  RFC2018: section 1, page 2, last paragraph
 			 */
 			tcplog("SACK Permitted unexpected, "
 			    "connection aborted");
@@ -1062,20 +1107,24 @@
 
 		/*
 		 * Initialize receive structure.
+		 * XXXAO: TODO
 		 */
 		tp->rcv_adv += rwin;	/* XXX */
 		tp->irs = th->th_seq;
 		tp->rcv_up = th->th_seq;
 		tcp_rcvseqinit(tp);
 
+		tcp_init_rcv(tp, seq);	/* XXXAO */
+		tcp_init_snd(tp, ack);	/* XXXAO */
+
 		/*
 		 * Process SYN and integrate sequence number.
+		 * XXXAO: TODO
 		 */
 		tp->snd_una++;
 		tp->snd_wu_seq = th->th_seq;
 		tp->snd_wu_ack = th->th_ack;
 		th->th_seq++;		/* SYN is acked */
-		thflags &= ~TH_SYN;	/* SYN is processed */
 
 		tp->t_starttime = ticks;
 		tp->t_state = TCPS_ESTABLISHED;
@@ -1089,8 +1138,12 @@
 
 		tcpstat.tcps_connects++;
 		break;
+
 	/*
 	 * All other states where a connection was established before.
+	 * The test are ordered by their knock-out precedence.
+	 * The simplest checks that fail a segment come first.
+	 * The logical ordering of RFC793 is still maintained.
 	 */
 	case TCPS_ESTABLISHED:
 	case TCPS_CLOSE_WAIT:
@@ -1100,12 +1153,15 @@
 	case TCPS_FIN_WAIT_2:
 		/*
 		 * SYN and RST are handled separately below.
+		 * RST takes precedence in all flag combinations.
+		 *  RFC793: section 3.9, page 70-71
 		 */
 		if (thflags & (TH_SYN|TH_RST))
 			break;
 
 		/*
 		 * Segments without ACK are invalid.
+		 *  RFC793: section 3.9, page 72, fifth check, first sentence
 		 */
 		if (!(thflags & TH_ACK)) {
 			tcplog("ACK missing, segment ignored");
@@ -1113,78 +1169,145 @@
 		}
 
 		/*
-		 * Don't accept ack'ing of older than previously ack'd data.
-		 * XXXAO: Careful with out-of-order data. Must check seq too.
-		 * reordering and bidirectional data transfer.
-		 * XXXAO: Is this check really useful?
+		 * Don't accept missing TS when TS was negotiated and
+		 * vice versa.
+		 *  RFC1323bis: section 3.2, last paragraph, last sentence
 		 */
-		if (SEQ_LT(th->th_ack, tp->snd_una) &&
-		    SEQ_LT(th->th_seq, tp->rcv_nxt)) {
-			tcplog("Acking old data, segment ignored, "
-			    "sending challenge ACK");
-			goto dropafterack;
+		if ((tp->t_flags & TF_TIMESTAMP) && !(to.to_flags & TOF_TS)) {
+			tcplog("Timestamp missing, segment ignored");
+			goto drop;
+		}
+		if (!(tp->t_flags & TF_TIMESTAMP) && (to.to_flags & TOF_TS)) {
+			tcplog("Timestamp unexpected, segment ignored");
+			goto drop;
 		}
 
 		/*
-		 * Don't accept ack'ing of more than actually sent data.
+		 * Don't accept SACK when it wasn't negotiated at
+		 * connection setup time.
+		 *  RFC2018: section 1, page 2, last paragraph
 		 */
-		if (SEQ_GT(th->th_ack, tp->snd_max)) {
-			tcplog("Acking data not yet sent, segment ignored, "
-			    "sending challenge ACK");
-			tcpstat.tcps_rcvacktoomuch++;
-			goto dropafterack;
+		if ((to.to_flags & TOF_SACK) &&
+		    !(tp->t_flags & TF_SACK_PERM)) {
+			tcplog("SACK unexpected, segment ignored");
+			goto drop;
 		}
 
 		/*
-		 * Don't accept start of SEQ beyond receive window.
-		 * Allow for a window probe with one byte.
-		 * XXXAO: Window probe statistics.
+		 * PAWS: Protection against wrapped sequence numbers.
+		 *
+		 * Don't accept remote ts older than already seen,
+		 * or reflected ts newer than what we send last.
+		 *
+		 * We store the receive time as uptime with second
+		 * resolution.  This makes us independent from the
+		 * wrap-around after 2^32 / hz (24.8 days at 1ms hz).
+		 *
+		 *  RFC1323bis: section 4.2.1 and 4.2.3
 		 */
-		if (SEQ_GT(th->th_seq, tp->rcv_nxt + tp->rcv_win)) {
-			tcplog("Data beyond window, segment ignored, "
-			    "sending challenge ACK");
-			goto dropafterack;
-		}
+		if (to.to_flags & TOF_TS) {
+			struct bintime bt;
 
-		/*
-		 * Don't accept too old retransmits.
-		 * XXXAO: Use largest window we've ever sent.
-		 * sb_hiwat is pretty much that.  We normally
-		 * don't shrink the receive socket buffer.
-		 */
-		if (SEQ_LT(th->th_seq,
-		    tp->rcv_nxt - so->so_rcv.sb_hiwat - tlen)) {
-			tcplog("Too old retransmit, segment ignored, "
-			    "sending challenge ACK");
-			goto dropafterack;
+			getbinuptime(&bt);
+			if (bt.sec - tp->t_rcvtime < ((tcp_ts)0x0 - 1) / hz) {
+				if (TSTMP_LT(to.to_tsval, tp->snd_tsecr) {
+					tcplog("Timestamp too old, "
+					    "sending challenge ack");
+					goto dropafterack;
+				}
+				if (TSTMP_GT(to.to_tsecr, tp->snd_tsval) {
+					tcplog("Timestamp too new, "
+					    "sending challenge ack");
+					goto dropafterack;
+				}
+			}
 		}
 
 		/*
-		 * Don't accept missing TS when TS was negotiated and
-		 * vice versa.
+		 * Validate the segment against the window.  We don't accept
+		 * segments with SEQ totally outside of receive window, unless
+		 * one of the special cases described below applies.  Any data
+		 * portion hanging outside the window to either side will be
+		 * chopped off later.
+		 *  RFC793: section 3.3, page 24-26
+		 *  RFC793: section 3.9, page 69, first check, the four cases
+		 *
+		 * Window probes are sent when we advertized a zero window
+		 * because the receive socket buffer is full.  When the
+		 * application reads enough data from it we send a window
+		 * update.  This is not reliable though and the remote host
+		 * periodically probes if we've got space again.  It does
+		 * this by sending one byte of data.  We let the through this
+		 * one byte, truncate it later and process the remainder of
+		 * the segment including any options.  This way timers are
+		 * updated properly.  TH_FIN is let through even on a zero
+		 * receive window.
+		 *  RFC793: section 3.7, page 42, seventh and eigth paragraph
+		 *  RFC1122: section 4.2.2.17
+		 *
+		 * Urgent data in segments must be processed even if the window
+		 * is partially or completely closed.  Any normal data will
+		 * be truncated and not processed depending on the current
+		 * size of rwin.
+		 *  RFC793: section 3.3, page 26, third paragraph
+		 *
+		 * A keepalive is sent to solicit a response on an otherwise
+		 * idle connection.  Two methods exist: send a segment with
+		 * seq=rcv_nxt-1 without data, or with exactly one byte of
+		 * (bogus) data.  Normally this segment would be rejected
+		 * and a resynchronization ACK be sent (as intended). However
+		 * we want to further process the segment to update a couple
+		 * of timers and to further look at possible TCP options.
+		 *  RFC1122: section 4.2.3.6
 		 */
-		if ((tp->t_flags & TF_TIMESTAMP) && !(to.to_flags & TOF_TS)) {
-			tcplog("Timestamp missing, segment ignored");
-			goto drop;
+		if (SEQ_LT(th->th_seq + tlen, tp->rcv_nxt) ||
+		    SEQ_GT(th->th_seq, tp->rcv_nxt + (rwin ? (rwin - 1) : rwin))) {
+			/*
+			 * The connection is idle and this
+			 * is a keepalive.
+			 */
+			if (th->th_seq == tp->rcv_nxt - 1 &&
+			    th->th_ack == tp->snd_nxt &&
+			    tlen <= 1 && !(thflags & TH_URG)) {
+				if (tlen == 0) {
+					th->th_seq = tp->rcv_nxt;
+					thflags &= ~TH_FIN;
+				}
+				tp->t_flags |= TF_ACKNOW;
+				tcps.tcps_rcv_keepalive++;
+			} else if ((thflags & TH_URG) && th->th_urg && tlen &&
+			    th->th_seq == tp->rcv_nxt) {
+				/* Continue. XXXAO: tighter check. */
+			} else {
+				tcplog("Data outside window, segment ignored, "
+				    "sending challenge ACK");
+				goto dropafterack;
+			}
+			/*
+			 * NB: Continue with segment to update
+			 * last received timestamp, to capture
+			 * window updates, ACKs and congestion
+			 * control algorithms.  Any data above
+			 * the window is truncated later.
+			 */
 		}
-		if (!(tp->t_flags & TF_TIMESTAMP) && (to.to_flags & TOF_TS)) {
-			tcplog("Timestamp unexpected, segment ignored");
-			goto drop;
-		}
 
 		/*
-		 * Don't accept remote ts older than already seen,
-		 * reflected ts newer than what we send last.
-		 *
-		 * TODO-AO:
-		 * PAWS
+		 * Don't accept ack'ing of more than actually sent data.
+		 * Neither accept segments with a too old ACK.
+		 *  RFC793: section 3.9, page 72, fifth check, ESTABLISHED STATE,
+		 *   first paragraph, last sentence
+		 *  tcpsecure: section 5.2, mitigation of blind data injection
+		 *  tcpsecure: section 5.2 changes this to just drop
+		 * XXXAO: why?
 		 */
-		if ((to.to_flags & TOF_TS) &&
-		    ticks - tp->t_rcvtime < PAWS &&
-		    (!TSTMP_LT(to.to_tsval, tp->snd_tsecr)) ||
-		     TSTMP_GT(to.to_tsecr, tp->snd_tsval))) {
-			tcplog("Timestamp too old or new, segment ignored");
-			goto drop;
+		if (SEQ_GEQ(th->th_ack, tp->snd_nxt) ||
+		    SEQ_LT(th->th_ack, tp->snd_una - tp->snd_maxwnd)) {
+			tcplog("Acking data not yet sent or too old, "
+			    "segment ignored, sending challenge ACK");
+			tcpstat.tcps_rcvacktoomuch++;
+			goto dropafterack;
+			/* goto drop; */
 		}
 
 		/*
@@ -1195,20 +1318,11 @@
 		 */
 		if (SEQ_GT(th->th_seq, tp->snd_lastack) &&
 		    SEQ_LT(th->th_seq, tp->rcv_nxt) {
-			tcplog("Received retransmit before we sent delayed ACK, no action");
+			tcplog("Received retransmit before we sent delayed ACK,"
+			    " no action");
 		}
 
-		/*
-		 * Don't accept SACK when is wasn't negotiated at
-		 * connection setup time.
-		 */
-		if ((to.to_flags & TOF_SACK) &&
-		    !(tp->t_flags & TF_SACK_PERM)) {
-			tcplog("SACK unexpected, segment ignored");
-			goto drop;
-		}
-
-		/* XXX: stats */
+		/* XXXAO: stats? */
 		break;
 
 	/*
@@ -1226,6 +1340,8 @@
 	 * expensive MD5 hash computation.
 	 * In SYN_RECEIVED case syncache verified the signature
 	 * already.
+	 *  RFC2385: section 2.0, 3.0
+	 * XXXAO: Make work.
 	 */
 	if ((tp->t_flags & TF_SIGNATURE) && notalreadydone) {
 		/* Copy signature and compare. */
@@ -1241,64 +1357,83 @@
 	/*
 	 * Fast path for ACK-only segments.
 	 */
-	if (tlen == 0 && (thflags & (TH_ACK|TH_RST|TH_SYN)) == TH_ACK)
+	if (tlen == 0 && (thflags & (TH_ACK|TH_RST|TH_SYN|TH_URG)) == TH_ACK)
 		goto doack;
 
 	/*
 	 * Handle SYN and RST flags for existing connections.
 	 *
-	 * NB: The SYN_SENT case has removed the SYN bit from thflags
-	 * if the segment was accepted.
+	 * NB: The SYN_SENT case has removed SYN from thflags.
 	 */
 	if (thflags & TH_RST) {
 		/*
-		 * Any RST after TCPS_SYN_SENT must NOT carry the ACK flag.
-		 * RFC 793 page 65, section SEGMENT ARRIVES.
+		 * Filter out what we determine NOT to be legitimate RST's.
 		 */
-		if (tp->t_state > TCPS_SYN_SENT &&
-		    (thflags & TH_ACK)) {
-			tcplog("RST with ACK invalid, segment ignored");
-			tcpstat.tcps_badrst++;
-			goto drop;
-		}
-		/*
-		 * Check if the sequence number is NOT acceptable to us.
-		 */
-		if (tp->t_state == TCPS_SYN_SENT) {
+		switch (tp->t_state) {
+		case TCPS_SYN_SENT:
 			/*
 			 * In TCPS_SYN_SENT the RST MUST carry the ACK flag.
+			 *  RFC793: section 3.9, page 66, first check
 			 */
 			if (!(thflags & TH_ACK)) {
 				tcplog("RST without ACK invalid, "
 				    "segment ignored");
 				goto drop;
 			}
-			if (th->th_ack != tp->snd_iss + 1) {
-				/*
-				 * XXX: Account for end of window
-				 * if we had data sent with SYN.
-				 */
+			/*
+			 * The ACK must be within what we sent but does
+			 * not have to ACK the SYN.
+			 *  RFC793: section 3.9, page 66, first check
+			 */
+			if (SEQ_LT(th->th_ack, tp->snd_una) ||
+			    SEQ_GT(th->th_ack, th->snd_nxt)) {
 				tcplog("RST does not match, segment ignored");
 				tcpstat.tcps_badrst++;
 				goto drop;
 			}
-		} else if (tcp_insecure_rst == 0 &&
-		    (SEQ_DELTA(th->th_seq, tp_rcv_nxt) > 1 ||
-		     SEQ_DELTA(th->th_seq, tp_snd_last_ack) > 1)) {
-			tcplog("RST does not match (secure), segment ignored");
-			tcpstat.tcps_badrst++;
-			goto drop;
-		} else if (tcp_insecure_rst == 1 &&
-		    (SEQ_LT(th->th_seq, tp->snd_last_ack - 1) ||
-		     SEQ_GT(th->th_seq, tp->snd_last_ack + rwin))) {
-			tcplog("RST does not match (insecure), segment ignored");
-			tcpstat.tcps_badrst++;
-			goto drop;
+			break;
+
+		case TCPS_ESTABLISHED:
+		case TCPS_FIN_WAIT_1:
+		case TCPS_FIN_WAIT_2:
+		case TCPS_CLOSE_WAIT:
+		case TCPS_CLOSING:
+		case TCPS_LAST_ACK:
+			/*
+			 * Any RST after TCPS_SYN_SENT must NOT carry the ACK flag.
+			 *  RFC 793: page 65
+			 */
+			if (thflags & TH_ACK) {
+				tcplog("RST with ACK invalid, segment ignored");
+				tcpstat.tcps_badrst++;
+				goto drop;
+			}
+			/*
+			 * Check if the sequence number is NOT acceptable to us.
+			 *  RFC793: page 70, second check
+			 *  RFC4953: section 3.1.2 (discussion of various methods)
+			 * XXXAO: Three points: rcv_nxt, snd_last_ack, rcv_nxt+rwin?
+			 * XXXAO: Description of +-1 variance.
+			 */
+			if (tcp_secure_rst &&
+			    (SEQ_DELTA(th->th_seq, tp->rcv_nxt) > 1 ||
+			     SEQ_DELTA(th->th_seq, tp->snd_last_ack) > 1)) {
+				tcplog("RST does not match (secure), segment ignored");
+				tcpstat.tcps_badrst++;
+				goto drop;
+			} else if (!tcp_secure_rst &&
+			    (SEQ_LT(th->th_seq, tp->snd_last_ack - 1) ||
+			     SEQ_GT(th->th_seq, tp->snd_last_ack + rwin))) {
+				tcplog("RST does not match (insecure), segment ignored");
+				tcpstat.tcps_badrst++;
+				goto drop;
+			}
+			break;
 		}
 		tcplog("RST received, closing connection");
 
 		/*
-		 * Treat the different states appropriately.
+		 * Unwind the connection according to its state.
 		 */
 		switch (tp->t_state) {
 		case TCPS_SYN_SENT:
@@ -1306,54 +1441,82 @@
 		case TCPS_FIN_WAIT_1:
 		case TCPS_FIN_WAIT_2:
 		case TCPS_CLOSE_WAIT:
+			/*
+			 * The socket is still around and
+			 * we have to inform the application.
+			 *  RFC1122: section 4.2.2.12 [RST cause]
+			 *  RFC1122: section 4.2.2.13
+			 */
 			so->so_error = ECONNRESET;
+			/* XXXAO: Macro encapsulating state transition changes? */
 			tp->t_state = TCPS_CLOSED;
 			tcpstat.tcps_drops++;
 			tp = tcp_close(tp);
 			break;
+
 		case TCPS_CLOSING:
 		case TCPS_LAST_ACK:
+			/*
+			 * The socket is already gone,
+			 * just clean up and be done.
+			 */
 			tp = tcp_close(tp);
 			break;
 		}
 		goto drop;
 	} else if (thflags & TH_SYN) {
 		/*
+		 * We may get a SYN in these cases:
+		 *  remote host went down and comes back up
+		 *  retransmitted SYN-ACK when our ACK was lost
+		 *  malicous SYN
+		 *  simultaneous open in SYN_SENT case is handled there
+		 *
 		 * Instead of dropping the connection right away
 		 * we send a challenge ACK back.  If the connection
 		 * is dead or retried we get back a proper RST which
 		 * will be validated and close the connection then.
 		 * tcp-secure modified recommends behavior to protect
 		 * against "blind in the window" attacks.
+		 *  tcpsecure: section 4.2
 		 */
-		tcplog("SYN received, segment ignored, sending challenge ACK");
+		tcplog("SYN received, segment ignored, "
+		    "sending challenge ACK");
 		goto dropafterack;
 	}
 
 	/*
-	 * From here we start looking at the segment content and
-	 * the meat is real.
+	 * From here we start looking at the segment content.
+	 * Assert any assumption the code from this point on makes.
 	 */
 	KASSERT((thflags & (TH_SYN|TH_RST|TH_ACK)) == TH_ACK,
-	    ("%s: no ACK flag", __func__));
+	    ("%s: no ACK flag or TH_SYN, TH_RST present", __func__));
+	KASSERT(SEQ_GEQ(th->th_seq + tlen, tp->rcv_nxt),
+	    ("%s: th_seq+tlen < rcv_nxt", __func__));
+	KASSERT(SEQ_LT(th->th_seq, tp->rcv_nxt + rwin),
+	    ("%s: th_seq >= rcv_nxt+rwin", __func__));
 
 	/*
 	 * Trim segment on the left, i.e. it starts before rcv_nxt.
+	 *
+	 * XXXAO: we may have a zero window but have to process URG data.
+	 * Trim off all normal payload and leave only the urg data intact.
 	 */
 	todrop = tp->rcv_nxt - th->th_seq;
 	if (todrop > 0) {
-		if (todrop > tlen)
-			todrop = tlen;
+		KASSERT(todrop <= tlen,
+		    ("%s: left todrop > tlen", __func__));
 		if (todrop == tlen) {
 			/*
 			 * Any valid FIN must be to the left of the window.
 			 * At this point the FIN must be a duplicate or out
 			 * of sequence; drop it.
+			 * XXXAO: Fixup
 			 */
 			thflags &= ~TH_FIN;
 			/*
 			 * Send an ACK to resynchronize and drop any data.
-			 * But keep on processing for ACK.
+			 * But keep on processing for ACK and options.
 			 */
 			tp->t_flags |= TF_ACKNOW;
 			tcpstat.tcps_rcvduppack++;
@@ -1362,7 +1525,12 @@
 			tcpstat.tcps_rcvpartduppack++;
 			tcpstat.tcps_rcvpartdupbyte += todrop;
 		}
-		drop_hdrlen += todrop;	/* Drop from the front afterwards. */
+		/*
+		 * Drop from front later together with
+		 * delayed header drop and adjust segment
+		 * length variables.
+		 */
+		drop_hdrlen += todrop;
 		th->th_seq += todrop;
 		tlen -= todrop;
 		/*
@@ -1376,16 +1544,20 @@
 			th->th_urp = 0;
 		}
 	}
+
 	/*
 	 * Trim segment to the right, ie. it ends after window,
 	 * drop trailing data (and PUSH and FIN);
 	 * if nothing left, just ACK.
+	 * XXXAO: for window probe we may have to trim off the one byte.
 	 */
-	todrop = (th->th_seq + tlen) - (tp->rcv_nxt + rwin);
+	if (!TCPS_HAVERCVDFIN(tp->t_state))
+		todrop = th->th_seq + tlen, tp->rcv_nxt + rwin;
+	else
+		todrop = tlen;
 	if (todrop > 0) {
-		KASSERT(todrop <= tlen, ("%s: todrop > tlen", __func__));
-		if (todrop >= tlen) {
-			tcpstat.tcps_rcvbyteafterwin += tlen;
+		KASSERT(todrop <= tlen, ("%s: right todrop > tlen", __func__));
+		if (todrop == tlen) {
 			/*
 			 * If window is closed we can only take segments at
 			 * the window edge, and have to drop data and PUSH
@@ -1396,20 +1568,33 @@
 			if (rwin == 0 && th->th_seq == tp->rcv_nxt) {
 				tp->t_flags |= TF_ACKNOW;
 				tcpstat.tcps_rcvwinprobe++;
-			} else {
-				tcolog("");
-				goto dropafterack;
 			}
-			todrop = tlen;
-		} else
-			tcpstat.tcps_rcvbyteafterwin += todrop;
-		m_adj(m, -todrop);
+		}
+		m_adj(m, -todrop);		/* Drop from tail. */
 		tlen -= todrop;
 		thflags &= ~(TH_PUSH|TH_FIN);
 		tcpstat.tcps_rcvpackafterwin++;
-		/* XXX: Urgent pointer? */
+		tcpstat.tcps_rcvbyteafterwin += todrop;
+		/* XXXAO: Urgent pointer? */
+		if ((thflags & TH_URG) && th->th_urp > todrop)
+			th->th_urp -= todrop;
+		else if (thflags & TH_URG) {
+			thflags &= ~TH_URG;
+			th->th_urp = 0;
+		}
+	}
+
+	/*
+	 * Urgent pointer is invalid when we dont have any data.
+	 * XXXAO: Not really.  Urgent data is a hack.
+	 */
+	if ((thflags & TH_URG) && tlen == 0) {
+		thflags &= ~TH_URG;
+		th->th_urp = 0;
 	}
-	KASSERT(tlen >= 0, ("%s: tlen < 0", __func__));
+
+	KASSERT(tlen >= 0,
+	    ("%s: tlen < 0", __func__));
 
 	/*
 	 * If new data is received on a connection after the
@@ -1429,19 +1614,18 @@
 
 doack:
 	/*
-	 * Ack processing.
+	 * ACK, timing, option and congestion control processing.
 	 */
-	KASSERT(SEQ_GEQ(th->th_ack, tp->snd_una),
-	    ("%s: th_ack < snd_una", __func__));
-
 	acked = th->th_ack - tp->snd_una;
+	if (acked < 0)
+		acked = 0;
 	tcpstat.tcps_rcvackpack++;
 	tcpstat.tcps_rcvackbyte += acked;
 
 	/*
-	 * Update window information.
+	 * Update send window information.
 	 */
-	nudgeoutput = tcp_do_wu(tp, th, tiwin, tlen);
+	nudgeoutput = tcp_do_wu(tp, th, &to, tiwin, tlen);
 
 	/*
 	 * Update and recompute connection timing information.
@@ -1450,12 +1634,10 @@
 	tcp_do_time(tp, th, &to, acked, tlen);
 
 	/*
-	 * Update SACK information.
+	 * Update send SACK information.
 	 */
-	if (((tp->t_flags & TF_SACK) && (to.to_flags & TOF_SACK)) ||
-	    !TAILQ_EMPTY(&tp->snd_holes)) {
+	if ((to.to_flags & TOF_SACK) || !TAILQ_EMPTY(&tp->snd_holes))
 		tcp_sack_doack(tp, &to, th->th_ack);
-	}
 
 	/*
 	 * Update congestion control information.
@@ -1464,10 +1646,16 @@
 
 	/*
 	 * Drop acknowledged data from send socket buffer.
+	 *  RFC793: section 3.9, page 72, fifth check
 	 */
 	if (acked > 0)
+		int cantrcvmore;
+
 		SOCKBUF_LOCK(&so->so_snd);
-		KASSERT(acked + 1 <= so->so_snd.sb_cc,
+
+		KASSERT(SEQ_GT(th->th_ack, tp->snd_nxt),
+		    ("%s: ", __func__));
+		KASSERT(acked <= so->so_snd.sb_cc + 1,
 		    ("%s: more acked than in send buffer", __func__));
 
 		/*
@@ -1485,15 +1673,29 @@
 			tp->snd_wnd -= acked;
 			ourfinisacked = 0;
 		}
+
 		/*
 		 * Advance the unacknowledged pointer.
+		 *  RFC793: section 3.9, page 72, fifth check
 		 */
 		tp->snd_una = th->th_ack;
+
+		/*
+		 * Obtain sb_state before unlock for later use.
+		 */
+		cantrcvmore = so->so_rcv.sb_state & SBS_CANTRCVMORE;
+
 		/*
+		 * Wake up and inform any writers on the socket.
+		 *
 		 * NB: sowwakeup_locked() does an implicit unlock.
 		 */
 		sowwakeup_locked(so);
 
+		/*
+		 * When our FIN was ack'ed perform the appropriate
+		 * state transitions and release unnessary resources.
+		 */
 		if (ourfinisacked) {
 			KASSERT((tp->t_flags & TF_SENTFIN) &&
 			    tp->t_state > TCPS_CLOSE_WAIT &&
@@ -1501,21 +1703,22 @@
 			    ("%s: got ack for FIN but haven't sent FIN yet",
 			    __func__));
 
-			switch (tp->t_state) {
 			/*
-			 * In FIN_WAIT_1 state enter the FIN-WAIT-2 state.
-			 * Any transition to CLOSING happens later.
-			 * XXX: comment.
+			 * Handle ack'ed FIN according to previous state.
 			 */
+			switch (tp->t_state) {
 			case TCPS_FIN_WAIT_1:
 				/*
-				 * If we can't receive any more
-				 * data, then closing user can proceed.
-				 * Starting the timer is contrary to the
+				 * If we can't receive any more data,
+				 * then closing user can proceed.
+				 * XXXAO: better description and reference
+				 * to discussion.
+				 *
+				 * NB: Starting the timer is contrary to the
 				 * specification, but if we don't get a FIN
 				 * we'll hang forever.
 				 */
-				if (so->so_rcv.sb_state & SBS_CANTRCVMORE) {
+				if (cantrcvmore) {
 					soisdisconnected(so);
 					tcp_timer_activate(tp, TT_2MSL,
 					    (tcp_fast_finwait2_recycle ?
@@ -1526,12 +1729,12 @@
 				tp->t_state = TCPS_FIN_WAIT_2;
 				break;
 
-			/*
-			 * In CLOSING state enter the TIME-WAIT state.
-			 * tcp_twstart() discards this tcpcb and creates
-			 * a compressed state.
-			 */
 			case TCPS_CLOSING:
+				/*
+				 * Create a compressed TIME-WAIT state
+				 * with minimal information and discard
+				 * this tcpcb to save memory.
+				 */
 				tcp_twstart(tp);
 				tp = NULL;
 				INP_INFO_WUNLOCK(&tcbinfo);
@@ -1540,23 +1743,21 @@
 				goto done;
 				break;
 
-			/*
-			 * In LAST_ACK, we may still be waiting for data
-			 * to drain and/or to be acked, as well as for

>>> TRUNCATED FOR MAIL (1000 lines) <<<



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200805312002.m4VK2ZrH067296>