Received: by 2002:a25:8b91:0:0:0:0:0 with SMTP id j17csp4631670ybl; Wed, 22 Jan 2020 01:38:47 -0800 (PST) X-Google-Smtp-Source: APXvYqwyI7Qz7tTpP84gBbsZSuhSwkEY8qHKhmbUmUv603FLH9mgIQlQSeO17PUc9x5RooC9lwUv X-Received: by 2002:a05:6808:907:: with SMTP id w7mr6212735oih.137.1579685927199; Wed, 22 Jan 2020 01:38:47 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1579685927; cv=none; d=google.com; s=arc-20160816; b=rhOVS2cxs9Vs59KWVr6shaJmstHLP9gD/9sqCGK6Fo8tqX2GpzKHom0xdQMx6/3Qn/ erhAXSqogdwvH+YBskMzo2dtgPDhNYGjqyX/QzDpeVfRMCHI9/46x5PGi5BpJJ/qAcar CRsKD61FG3XodyoOMHs+lsMO3F5A9lL2kXmHjq+VYa5h1MyHQqomGZRK0ikYGfwm5r5f 6qaMJrTfdk9bvBCqGqJy3UtVckU3IfdhVc55PBLkoBGzLwJE1TqIYA5DCsAO1thfB/Dh y2YhQJ1r9FeaSLxrlEIEePJ0c62JfY+sy+/FLkvByCY1/USrCXdhUr4G8SaqtX7+BAbx 74og== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :user-agent:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=dHGz013aZUy6D5ugUtnqV7+NeeIrI17qxePFtzjub3g=; b=LkDEjVUl01MT3kqdHgB3/xCbwQ47dkPcU4md9Ni1bMwFytq6gIRltXzJbpOY7Zxfhs lC7aHLgfr/+O2q7nSrB9oOMIg2G6pfs9Q1uj/9GB2WRmGt046RzJ5hIB1FKfzu2v0g6P ERzar5XaYtKBOvXZMVfwod5mFxDUpnQBWXDv0ZzaSVg8dxn1IqCfI5YZrIWfQYEaa+Wh DInNuH8yJZDhUC4Tp5G2v8LiBhvix6cED5zzgrHYuLfpdSedcKc8yUANSV99CmgdPd87 2U3tSvP4Elt7JC0E+VOYqUrvLDQB5UABMfuneFa0bAD+ZEN4reBNKZvaXoJCr8JnVpn4 naEA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b="XOv9N/9P"; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id h203si21704948oif.3.2020.01.22.01.38.35; Wed, 22 Jan 2020 01:38:47 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b="XOv9N/9P"; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1732118AbgAVJgs (ORCPT + 99 others); Wed, 22 Jan 2020 04:36:48 -0500 Received: from mail.kernel.org ([198.145.29.99]:52342 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1731991AbgAVJgp (ORCPT ); Wed, 22 Jan 2020 04:36:45 -0500 Received: from localhost (83-86-89-107.cable.dynamic.v4.ziggo.nl [83.86.89.107]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by mail.kernel.org (Postfix) with ESMTPSA id CA6B424687; Wed, 22 Jan 2020 09:36:43 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=default; t=1579685804; bh=dFYKqZc6dYI9weUBdmlfJSBiQmCkdRyAKsj4+kb9y9A=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=XOv9N/9PPDFOFYMRwKtsl1nTxpTFT31chRkTGeEjCf+hHjjJ7zMNSig52E4zGtd+U 8ivHyz2Vbh0PueX2uHFMQIoFaLW0s3FnUDcLaBFMsrdWsgY60Jmx2MOHuwMObTIADC BF0biuOBULoUaGGFunpfZgAzoIAnRPKzxizlQYQQ= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Pengcheng Yang , Neal Cardwell , "David S. Miller" Subject: [PATCH 4.9 85/97] tcp: fix marked lost packets not being retransmitted Date: Wed, 22 Jan 2020 10:29:29 +0100 Message-Id: <20200122092809.980970083@linuxfoundation.org> X-Mailer: git-send-email 2.25.0 In-Reply-To: <20200122092755.678349497@linuxfoundation.org> References: <20200122092755.678349497@linuxfoundation.org> User-Agent: quilt/0.66 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Pengcheng Yang [ Upstream commit e176b1ba476cf36f723cfcc7a9e57f3cb47dec70 ] When the packet pointed to by retransmit_skb_hint is unlinked by ACK, retransmit_skb_hint will be set to NULL in tcp_clean_rtx_queue(). If packet loss is detected at this time, retransmit_skb_hint will be set to point to the current packet loss in tcp_verify_retransmit_hint(), then the packets that were previously marked lost but not retransmitted due to the restriction of cwnd will be skipped and cannot be retransmitted. To fix this, when retransmit_skb_hint is NULL, retransmit_skb_hint can be reset only after all marked lost packets are retransmitted (retrans_out >= lost_out), otherwise we need to traverse from tcp_rtx_queue_head in tcp_xmit_retransmit_queue(). Packetdrill to demonstrate: // Disable RACK and set max_reordering to keep things simple 0 `sysctl -q net.ipv4.tcp_recovery=0` +0 `sysctl -q net.ipv4.tcp_max_reordering=3` // Establish a connection +0 socket(..., SOCK_STREAM, IPPROTO_TCP) = 3 +0 setsockopt(3, SOL_SOCKET, SO_REUSEADDR, [1], 4) = 0 +0 bind(3, ..., ...) = 0 +0 listen(3, 1) = 0 +.1 < S 0:0(0) win 32792 +0 > S. 0:0(0) ack 1 <...> +.01 < . 1:1(0) ack 1 win 257 +0 accept(3, ..., ...) = 4 // Send 8 data segments +0 write(4, ..., 8000) = 8000 +0 > P. 1:8001(8000) ack 1 // Enter recovery and 1:3001 is marked lost +.01 < . 1:1(0) ack 1 win 257 +0 < . 1:1(0) ack 1 win 257 +0 < . 1:1(0) ack 1 win 257 // Retransmit 1:1001, now retransmit_skb_hint points to 1001:2001 +0 > . 1:1001(1000) ack 1 // 1001:2001 was ACKed causing retransmit_skb_hint to be set to NULL +.01 < . 1:1(0) ack 2001 win 257 // Now retransmit_skb_hint points to 4001:5001 which is now marked lost // BUG: 2001:3001 was not retransmitted +0 > . 2001:3001(1000) ack 1 Signed-off-by: Pengcheng Yang Acked-by: Neal Cardwell Tested-by: Neal Cardwell Signed-off-by: David S. Miller Signed-off-by: Greg Kroah-Hartman --- net/ipv4/tcp_input.c | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) --- a/net/ipv4/tcp_input.c +++ b/net/ipv4/tcp_input.c @@ -923,9 +923,10 @@ static void tcp_update_reordering(struct /* This must be called before lost_out is incremented */ static void tcp_verify_retransmit_hint(struct tcp_sock *tp, struct sk_buff *skb) { - if (!tp->retransmit_skb_hint || - before(TCP_SKB_CB(skb)->seq, - TCP_SKB_CB(tp->retransmit_skb_hint)->seq)) + if ((!tp->retransmit_skb_hint && tp->retrans_out >= tp->lost_out) || + (tp->retransmit_skb_hint && + before(TCP_SKB_CB(skb)->seq, + TCP_SKB_CB(tp->retransmit_skb_hint)->seq))) tp->retransmit_skb_hint = skb; if (!tp->lost_out ||