aboutsummaryrefslogtreecommitdiffstats
path: root/net
diff options
context:
space:
mode:
authorAlexey Kodanev <alexey.kodanev@oracle.com>2017-02-22 05:23:55 -0500
committerDavid S. Miller <davem@davemloft.net>2017-02-22 16:35:32 -0500
commit00355fa5bb89840c48eb11eb6d84c6c3b128a839 (patch)
tree544ae213a42d76846ecbf01f87bd404dee819312 /net
parentec7cb62d18d854ea09df8b7194e7e710985f8b9a (diff)
tcp: setup timestamp offset when write_seq already set
Found that when randomized tcp offsets are enabled (by default) TCP client can still start new connections without them. Later, if server does active close and re-uses sockets in TIME-WAIT state, new SYN from client can be rejected on PAWS check inside tcp_timewait_state_process(), because either tw_ts_recent or rcv_tsval doesn't really have an offset set. Here is how to reproduce it with LTP netstress tool: netstress -R 1 & netstress -H 127.0.0.1 -lr 1000000 -a1 [...] < S seq 1956977072 win 43690 TS val 295618 ecr 459956970 > . ack 1956911535 win 342 TS val 459967184 ecr 1547117608 < R seq 1956911535 win 0 length 0 +1. < S seq 1956977072 win 43690 TS val 296640 ecr 459956970 > S. seq 657450664 ack 1956977073 win 43690 TS val 459968205 ecr 296640 Fixes: 95a22caee396 ("tcp: randomize tcp timestamp offsets for each connection") Signed-off-by: Alexey Kodanev <alexey.kodanev@oracle.com> Acked-by: Eric Dumazet <edumazet@google.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net')
-rw-r--r--net/ipv4/tcp_ipv4.c16
-rw-r--r--net/ipv6/tcp_ipv6.c16
2 files changed, 20 insertions, 12 deletions
diff --git a/net/ipv4/tcp_ipv4.c b/net/ipv4/tcp_ipv4.c
index 8c124d4ef4b7..9a89b8deafae 100644
--- a/net/ipv4/tcp_ipv4.c
+++ b/net/ipv4/tcp_ipv4.c
@@ -145,6 +145,7 @@ int tcp_v4_connect(struct sock *sk, struct sockaddr *uaddr, int addr_len)
145 struct flowi4 *fl4; 145 struct flowi4 *fl4;
146 struct rtable *rt; 146 struct rtable *rt;
147 int err; 147 int err;
148 u32 seq;
148 struct ip_options_rcu *inet_opt; 149 struct ip_options_rcu *inet_opt;
149 struct inet_timewait_death_row *tcp_death_row = &sock_net(sk)->ipv4.tcp_death_row; 150 struct inet_timewait_death_row *tcp_death_row = &sock_net(sk)->ipv4.tcp_death_row;
150 151
@@ -234,12 +235,15 @@ int tcp_v4_connect(struct sock *sk, struct sockaddr *uaddr, int addr_len)
234 sk_setup_caps(sk, &rt->dst); 235 sk_setup_caps(sk, &rt->dst);
235 rt = NULL; 236 rt = NULL;
236 237
237 if (!tp->write_seq && likely(!tp->repair)) 238 if (likely(!tp->repair)) {
238 tp->write_seq = secure_tcp_sequence_number(inet->inet_saddr, 239 seq = secure_tcp_sequence_number(inet->inet_saddr,
239 inet->inet_daddr, 240 inet->inet_daddr,
240 inet->inet_sport, 241 inet->inet_sport,
241 usin->sin_port, 242 usin->sin_port,
242 &tp->tsoffset); 243 &tp->tsoffset);
244 if (!tp->write_seq)
245 tp->write_seq = seq;
246 }
243 247
244 inet->inet_id = tp->write_seq ^ jiffies; 248 inet->inet_id = tp->write_seq ^ jiffies;
245 249
diff --git a/net/ipv6/tcp_ipv6.c b/net/ipv6/tcp_ipv6.c
index 21c719965b6b..60a5295a7de6 100644
--- a/net/ipv6/tcp_ipv6.c
+++ b/net/ipv6/tcp_ipv6.c
@@ -122,6 +122,7 @@ static int tcp_v6_connect(struct sock *sk, struct sockaddr *uaddr,
122 struct flowi6 fl6; 122 struct flowi6 fl6;
123 struct dst_entry *dst; 123 struct dst_entry *dst;
124 int addr_type; 124 int addr_type;
125 u32 seq;
125 int err; 126 int err;
126 struct inet_timewait_death_row *tcp_death_row = &sock_net(sk)->ipv4.tcp_death_row; 127 struct inet_timewait_death_row *tcp_death_row = &sock_net(sk)->ipv4.tcp_death_row;
127 128
@@ -285,12 +286,15 @@ static int tcp_v6_connect(struct sock *sk, struct sockaddr *uaddr,
285 286
286 sk_set_txhash(sk); 287 sk_set_txhash(sk);
287 288
288 if (!tp->write_seq && likely(!tp->repair)) 289 if (likely(!tp->repair)) {
289 tp->write_seq = secure_tcpv6_sequence_number(np->saddr.s6_addr32, 290 seq = secure_tcpv6_sequence_number(np->saddr.s6_addr32,
290 sk->sk_v6_daddr.s6_addr32, 291 sk->sk_v6_daddr.s6_addr32,
291 inet->inet_sport, 292 inet->inet_sport,
292 inet->inet_dport, 293 inet->inet_dport,
293 &tp->tsoffset); 294 &tp->tsoffset);
295 if (!tp->write_seq)
296 tp->write_seq = seq;
297 }
294 298
295 if (tcp_fastopen_defer_connect(sk, &err)) 299 if (tcp_fastopen_defer_connect(sk, &err))
296 return err; 300 return err;