summaryrefslogtreecommitdiffstats
path: root/sys/netinet/tcp_syncache.c
diff options
context:
space:
mode:
authorjlemon <jlemon@FreeBSD.org>2002-06-28 19:12:38 +0000
committerjlemon <jlemon@FreeBSD.org>2002-06-28 19:12:38 +0000
commit3d67b562831b016305d6434fbec32de963646692 (patch)
treefba3f91110d01f6c448f415bf200a07f462b1c4b /sys/netinet/tcp_syncache.c
parent9fbb8d216f8792d780e5664546c331feb5a73f98 (diff)
downloadFreeBSD-src-3d67b562831b016305d6434fbec32de963646692.zip
FreeBSD-src-3d67b562831b016305d6434fbec32de963646692.tar.gz
One possible code path for syncache_respond() is:
syncache_respond(A), ip_output(), ip_input(), tcp_input(), syncache_badack(B) Which winds up deleting a different entry from the syncache. Handle this by not utilizing the next entry in the timer chain until after syncache_respond() completes. The case of A == B should not be possible. Problem found by: Don Bowman <don@sandvine.com>
Diffstat (limited to 'sys/netinet/tcp_syncache.c')
-rw-r--r--sys/netinet/tcp_syncache.c8
1 files changed, 7 insertions, 1 deletions
diff --git a/sys/netinet/tcp_syncache.c b/sys/netinet/tcp_syncache.c
index 0ef2f3d..a261d21 100644
--- a/sys/netinet/tcp_syncache.c
+++ b/sys/netinet/tcp_syncache.c
@@ -372,19 +372,25 @@ syncache_timer(xslot)
if (ticks < nsc->sc_rxttime)
break;
sc = nsc;
- nsc = TAILQ_NEXT(sc, sc_timerq);
inp = sc->sc_tp->t_inpcb;
INP_LOCK(inp);
if (slot == SYNCACHE_MAXREXMTS ||
slot >= tcp_syncache.rexmt_limit ||
inp->inp_gencnt != sc->sc_inp_gencnt) {
+ nsc = TAILQ_NEXT(sc, sc_timerq);
syncache_drop(sc, NULL);
tcpstat.tcps_sc_stale++;
INP_UNLOCK(inp);
continue;
}
+ /*
+ * syncache_respond() may call back into the syncache to
+ * to modify another entry, so do not obtain the next
+ * entry on the timer chain until it has completed.
+ */
(void) syncache_respond(sc, NULL);
INP_UNLOCK(inp);
+ nsc = TAILQ_NEXT(sc, sc_timerq);
tcpstat.tcps_sc_retransmitted++;
TAILQ_REMOVE(&tcp_syncache.timerq[slot], sc, sc_timerq);
SYNCACHE_TIMEOUT(sc, slot + 1);
OpenPOWER on IntegriCloud