blob: aeec09c621504a7c875db831d0989ecf034f7a18 [file] [log] [blame]
QUICHE teama6ef0a62019-03-07 20:34:33 -05001// Copyright 2013 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "net/third_party/quiche/src/quic/core/quic_sent_packet_manager.h"
6
7#include <algorithm>
vasilvv872e7a32019-03-12 16:42:44 -07008#include <string>
QUICHE teama6ef0a62019-03-07 20:34:33 -05009
10#include "net/third_party/quiche/src/quic/core/congestion_control/general_loss_algorithm.h"
11#include "net/third_party/quiche/src/quic/core/congestion_control/pacing_sender.h"
12#include "net/third_party/quiche/src/quic/core/crypto/crypto_protocol.h"
dschinazi56fb53e2019-06-21 15:30:04 -070013#include "net/third_party/quiche/src/quic/core/proto/cached_network_parameters_proto.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050014#include "net/third_party/quiche/src/quic/core/quic_connection_stats.h"
wub967ba572019-04-01 09:27:52 -070015#include "net/third_party/quiche/src/quic/core/quic_types.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050016#include "net/third_party/quiche/src/quic/core/quic_utils.h"
17#include "net/third_party/quiche/src/quic/platform/api/quic_bug_tracker.h"
18#include "net/third_party/quiche/src/quic/platform/api/quic_flag_utils.h"
19#include "net/third_party/quiche/src/quic/platform/api/quic_flags.h"
20#include "net/third_party/quiche/src/quic/platform/api/quic_logging.h"
21#include "net/third_party/quiche/src/quic/platform/api/quic_map_util.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050022
23namespace quic {
24
25namespace {
26static const int64_t kDefaultRetransmissionTimeMs = 500;
27static const int64_t kMaxRetransmissionTimeMs = 60000;
28// Maximum number of exponential backoffs used for RTO timeouts.
29static const size_t kMaxRetransmissions = 10;
30// Maximum number of packets retransmitted upon an RTO.
31static const size_t kMaxRetransmissionsOnTimeout = 2;
32// The path degrading delay is the sum of this number of consecutive RTO delays.
33const size_t kNumRetransmissionDelaysForPathDegradingDelay = 2;
34
fayangb59c6f12020-03-23 15:06:14 -070035// The blachkhole delay is the sum of this number of consecutive RTO delays.
36const size_t kNumRetransmissionDelaysForBlackholeDelay = 5;
37
QUICHE teama6ef0a62019-03-07 20:34:33 -050038// Ensure the handshake timer isnt't faster than 10ms.
39// This limits the tenth retransmitted packet to 10s after the initial CHLO.
40static const int64_t kMinHandshakeTimeoutMs = 10;
41
42// Sends up to two tail loss probes before firing an RTO,
43// per draft RFC draft-dukkipati-tcpm-tcp-loss-probe.
44static const size_t kDefaultMaxTailLossProbes = 2;
45
QUICHE teama6ef0a62019-03-07 20:34:33 -050046// Returns true of retransmissions of the specified type should retransmit
47// the frames directly (as opposed to resulting in a loss notification).
48inline bool ShouldForceRetransmission(TransmissionType transmission_type) {
49 return transmission_type == HANDSHAKE_RETRANSMISSION ||
50 transmission_type == TLP_RETRANSMISSION ||
51 transmission_type == PROBING_RETRANSMISSION ||
fayang73d0ac42019-10-31 12:45:31 -070052 transmission_type == RTO_RETRANSMISSION ||
53 transmission_type == PTO_RETRANSMISSION;
QUICHE teama6ef0a62019-03-07 20:34:33 -050054}
55
fayang4ba55982019-05-13 05:53:22 -070056// If pacing rate is accurate, > 2 burst token is not likely to help first ACK
57// to arrive earlier, and overly large burst token could cause incast packet
58// losses.
59static const uint32_t kConservativeUnpacedBurst = 2;
60
QUICHE teama6ef0a62019-03-07 20:34:33 -050061} // namespace
62
63#define ENDPOINT \
64 (unacked_packets_.perspective() == Perspective::IS_SERVER ? "Server: " \
65 : "Client: ")
66
67QuicSentPacketManager::QuicSentPacketManager(
68 Perspective perspective,
69 const QuicClock* clock,
QUICHE team73957f12019-04-18 16:21:52 -070070 QuicRandom* random,
QUICHE teama6ef0a62019-03-07 20:34:33 -050071 QuicConnectionStats* stats,
fayang3be090d2020-02-11 14:05:08 -080072 CongestionControlType congestion_control_type)
QUICHE teama6ef0a62019-03-07 20:34:33 -050073 : unacked_packets_(perspective),
74 clock_(clock),
QUICHE team73957f12019-04-18 16:21:52 -070075 random_(random),
QUICHE teama6ef0a62019-03-07 20:34:33 -050076 stats_(stats),
77 debug_delegate_(nullptr),
78 network_change_visitor_(nullptr),
79 initial_congestion_window_(kInitialCongestionWindow),
wub37ec5962019-04-03 09:02:51 -070080 loss_algorithm_(GetInitialLossAlgorithm()),
QUICHE teama6ef0a62019-03-07 20:34:33 -050081 consecutive_rto_count_(0),
82 consecutive_tlp_count_(0),
83 consecutive_crypto_retransmission_count_(0),
84 pending_timer_transmission_count_(0),
85 max_tail_loss_probes_(kDefaultMaxTailLossProbes),
86 max_rto_packets_(kMaxRetransmissionsOnTimeout),
87 enable_half_rtt_tail_loss_probe_(false),
88 using_pacing_(false),
89 use_new_rto_(false),
90 conservative_handshake_retransmits_(false),
91 min_tlp_timeout_(
92 QuicTime::Delta::FromMilliseconds(kMinTailLossProbeTimeoutMs)),
93 min_rto_timeout_(
94 QuicTime::Delta::FromMilliseconds(kMinRetransmissionTimeMs)),
QUICHE teama6ef0a62019-03-07 20:34:33 -050095 largest_mtu_acked_(0),
fayang63a19842020-01-23 02:51:28 -080096 handshake_finished_(false),
fkastenholz59c653b2019-07-15 09:55:53 -070097 peer_max_ack_delay_(
QUICHE teama6ef0a62019-03-07 20:34:33 -050098 QuicTime::Delta::FromMilliseconds(kDefaultDelayedAckTimeMs)),
99 rtt_updated_(false),
QUICHE team9929cc42019-03-13 08:17:43 -0700100 acked_packets_iter_(last_ack_frame_.packets.rbegin()),
fayang62f867a2019-08-22 12:05:01 -0700101 pto_enabled_(false),
fayangce0a3162019-08-15 09:05:36 -0700102 max_probe_packets_per_pto_(2),
103 consecutive_pto_count_(0),
fayang19d2d5b2019-09-11 14:22:03 -0700104 handshake_mode_disabled_(false),
fayang4c908f02019-11-01 07:26:17 -0700105 skip_packet_number_for_pto_(false),
106 always_include_max_ack_delay_for_pto_timeout_(true),
fayang7085a6d2019-11-04 07:03:57 -0800107 pto_exponential_backoff_start_point_(0),
fayang75db4342019-11-04 13:29:14 -0800108 pto_rttvar_multiplier_(4),
fayang4aa22402020-01-07 11:36:07 -0800109 num_tlp_timeout_ptos_(0),
fayang9a0e1bd2020-02-19 13:13:04 -0800110 one_rtt_packet_acked_(false),
fayang2ccfbcf2020-02-28 12:37:08 -0800111 one_rtt_packet_sent_(false),
fayangc2fd6102020-03-02 17:27:38 -0800112 first_pto_srtt_multiplier_(0),
113 use_standard_deviation_for_pto_(false) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500114 SetSendAlgorithm(congestion_control_type);
115}
116
wub37ec5962019-04-03 09:02:51 -0700117LossDetectionInterface* QuicSentPacketManager::GetInitialLossAlgorithm() {
fayangbf3d2862019-06-20 14:13:44 -0700118 return &uber_loss_algorithm_;
wub37ec5962019-04-03 09:02:51 -0700119}
120
QUICHE teama6ef0a62019-03-07 20:34:33 -0500121QuicSentPacketManager::~QuicSentPacketManager() {}
122
123void QuicSentPacketManager::SetFromConfig(const QuicConfig& config) {
124 const Perspective perspective = unacked_packets_.perspective();
125 if (config.HasReceivedInitialRoundTripTimeUs() &&
126 config.ReceivedInitialRoundTripTimeUs() > 0) {
127 if (!config.HasClientSentConnectionOption(kNRTT, perspective)) {
128 SetInitialRtt(QuicTime::Delta::FromMicroseconds(
129 config.ReceivedInitialRoundTripTimeUs()));
130 }
131 } else if (config.HasInitialRoundTripTimeUsToSend() &&
132 config.GetInitialRoundTripTimeUsToSend() > 0) {
133 SetInitialRtt(QuicTime::Delta::FromMicroseconds(
134 config.GetInitialRoundTripTimeUsToSend()));
135 }
ianswett43eefae2019-08-02 12:27:15 -0700136 if (config.HasReceivedMaxAckDelayMs()) {
137 peer_max_ack_delay_ =
138 QuicTime::Delta::FromMilliseconds(config.ReceivedMaxAckDelayMs());
139 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500140 if (config.HasClientSentConnectionOption(kMAD0, perspective)) {
141 rtt_stats_.set_ignore_max_ack_delay(true);
142 }
143 if (config.HasClientSentConnectionOption(kMAD1, perspective)) {
ianswett43eefae2019-08-02 12:27:15 -0700144 rtt_stats_.set_initial_max_ack_delay(peer_max_ack_delay_);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500145 }
ianswettad65ab92019-10-28 07:19:07 -0700146 if (config.HasClientSentConnectionOption(kMAD2, perspective)) {
147 // Set the minimum to the alarm granularity.
ianswett8f90e512019-12-18 10:50:27 -0800148 min_tlp_timeout_ = kAlarmGranularity;
ianswettad65ab92019-10-28 07:19:07 -0700149 }
150 if (config.HasClientSentConnectionOption(kMAD3, perspective)) {
151 // Set the minimum to the alarm granularity.
ianswett8f90e512019-12-18 10:50:27 -0800152 min_rto_timeout_ = kAlarmGranularity;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500153 }
154
fayang961eaa02020-01-10 13:04:22 -0800155 if (config.HasClientSentConnectionOption(k2PTO, perspective)) {
156 pto_enabled_ = true;
157 }
158 if (config.HasClientSentConnectionOption(k1PTO, perspective)) {
159 pto_enabled_ = true;
160 max_probe_packets_per_pto_ = 1;
fayangce0a3162019-08-15 09:05:36 -0700161 }
162
fayang91ab42f2020-01-13 17:18:48 -0800163 if (config.HasClientSentConnectionOption(kPTOS, perspective)) {
fayang4c908f02019-11-01 07:26:17 -0700164 if (!pto_enabled_) {
165 QUIC_PEER_BUG
166 << "PTO is not enabled when receiving PTOS connection option.";
167 pto_enabled_ = true;
168 max_probe_packets_per_pto_ = 1;
169 }
170 skip_packet_number_for_pto_ = true;
171 }
172
fayang7085a6d2019-11-04 07:03:57 -0800173 if (pto_enabled_) {
174 if (config.HasClientSentConnectionOption(kPTOA, perspective)) {
fayang7085a6d2019-11-04 07:03:57 -0800175 always_include_max_ack_delay_for_pto_timeout_ = false;
176 }
177 if (config.HasClientSentConnectionOption(kPEB1, perspective)) {
fayang7085a6d2019-11-04 07:03:57 -0800178 StartExponentialBackoffAfterNthPto(1);
179 }
180 if (config.HasClientSentConnectionOption(kPEB2, perspective)) {
fayang7085a6d2019-11-04 07:03:57 -0800181 StartExponentialBackoffAfterNthPto(2);
182 }
fayang75db4342019-11-04 13:29:14 -0800183 if (config.HasClientSentConnectionOption(kPVS1, perspective)) {
fayang75db4342019-11-04 13:29:14 -0800184 pto_rttvar_multiplier_ = 2;
185 }
fayang4aa22402020-01-07 11:36:07 -0800186 if (config.HasClientSentConnectionOption(kPAG1, perspective)) {
187 QUIC_CODE_COUNT(one_aggressive_pto);
188 num_tlp_timeout_ptos_ = 1;
189 }
190 if (config.HasClientSentConnectionOption(kPAG2, perspective)) {
191 QUIC_CODE_COUNT(two_aggressive_ptos);
192 num_tlp_timeout_ptos_ = 2;
193 }
fayangc5d9b2c2020-03-02 11:12:26 -0800194 if (GetQuicReloadableFlag(quic_arm_pto_with_earliest_sent_time)) {
ianswett3b801132020-03-12 14:57:34 -0700195 if (config.HasClientSentConnectionOption(kPLE1, perspective) ||
196 config.HasClientSentConnectionOption(kTLPR, perspective)) {
fayangc5d9b2c2020-03-02 11:12:26 -0800197 QUIC_RELOADABLE_FLAG_COUNT_N(quic_arm_pto_with_earliest_sent_time, 1,
198 2);
199 first_pto_srtt_multiplier_ = 0.5;
200 } else if (config.HasClientSentConnectionOption(kPLE2, perspective)) {
201 QUIC_RELOADABLE_FLAG_COUNT_N(quic_arm_pto_with_earliest_sent_time, 2,
202 2);
203 first_pto_srtt_multiplier_ = 1.5;
204 }
fayang2ccfbcf2020-02-28 12:37:08 -0800205 }
fayangc2fd6102020-03-02 17:27:38 -0800206 if (GetQuicReloadableFlag(quic_use_standard_deviation_for_pto) &&
207 config.HasClientSentConnectionOption(kPSDA, perspective)) {
208 QUIC_RELOADABLE_FLAG_COUNT(quic_use_standard_deviation_for_pto);
209 use_standard_deviation_for_pto_ = true;
210 rtt_stats_.EnableStandardDeviationCalculation();
211 }
fayang4c908f02019-11-01 07:26:17 -0700212 }
213
QUICHE teama6ef0a62019-03-07 20:34:33 -0500214 // Configure congestion control.
215 if (config.HasClientRequestedIndependentOption(kTBBR, perspective)) {
216 SetSendAlgorithm(kBBR);
217 }
wuba9a43cb2019-07-17 15:22:42 -0700218 if (GetQuicReloadableFlag(quic_allow_client_enabled_bbr_v2) &&
219 config.HasClientRequestedIndependentOption(kB2ON, perspective)) {
220 QUIC_RELOADABLE_FLAG_COUNT(quic_allow_client_enabled_bbr_v2);
221 SetSendAlgorithm(kBBRv2);
222 }
223
QUICHE teama6ef0a62019-03-07 20:34:33 -0500224 if (config.HasClientRequestedIndependentOption(kRENO, perspective)) {
225 SetSendAlgorithm(kRenoBytes);
226 } else if (config.HasClientRequestedIndependentOption(kBYTE, perspective) ||
227 (GetQuicReloadableFlag(quic_default_to_bbr) &&
228 config.HasClientRequestedIndependentOption(kQBIC, perspective))) {
229 SetSendAlgorithm(kCubicBytes);
230 } else if (GetQuicReloadableFlag(quic_enable_pcc3) &&
231 config.HasClientRequestedIndependentOption(kTPCC, perspective)) {
232 SetSendAlgorithm(kPCC);
233 }
wuba9a43cb2019-07-17 15:22:42 -0700234
QUICHE teama6ef0a62019-03-07 20:34:33 -0500235 // Initial window.
236 if (GetQuicReloadableFlag(quic_unified_iw_options)) {
237 if (config.HasClientRequestedIndependentOption(kIW03, perspective)) {
238 initial_congestion_window_ = 3;
239 send_algorithm_->SetInitialCongestionWindowInPackets(3);
240 }
241 if (config.HasClientRequestedIndependentOption(kIW10, perspective)) {
242 initial_congestion_window_ = 10;
243 send_algorithm_->SetInitialCongestionWindowInPackets(10);
244 }
245 if (config.HasClientRequestedIndependentOption(kIW20, perspective)) {
246 initial_congestion_window_ = 20;
247 send_algorithm_->SetInitialCongestionWindowInPackets(20);
248 }
249 if (config.HasClientRequestedIndependentOption(kIW50, perspective)) {
250 initial_congestion_window_ = 50;
251 send_algorithm_->SetInitialCongestionWindowInPackets(50);
252 }
253 }
fayang402103c2020-01-10 13:36:24 -0800254 if (GetQuicReloadableFlag(quic_bbr_mitigate_overly_large_bandwidth_sample) &&
255 config.HasClientRequestedIndependentOption(kBWS5, perspective)) {
256 initial_congestion_window_ = 10;
257 send_algorithm_->SetInitialCongestionWindowInPackets(10);
258 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500259
danzh88e3e052019-06-13 11:47:18 -0700260 using_pacing_ = !GetQuicFlag(FLAGS_quic_disable_pacing_for_perf_tests);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500261
QUICHE teama6ef0a62019-03-07 20:34:33 -0500262 if (config.HasClientSentConnectionOption(kNTLP, perspective)) {
263 max_tail_loss_probes_ = 0;
264 }
265 if (config.HasClientSentConnectionOption(k1TLP, perspective)) {
266 max_tail_loss_probes_ = 1;
267 }
268 if (config.HasClientSentConnectionOption(k1RTO, perspective)) {
269 max_rto_packets_ = 1;
270 }
271 if (config.HasClientSentConnectionOption(kTLPR, perspective)) {
272 enable_half_rtt_tail_loss_probe_ = true;
273 }
274 if (config.HasClientSentConnectionOption(kNRTO, perspective)) {
275 use_new_rto_ = true;
276 }
277 // Configure loss detection.
fayanga590b752020-01-13 06:37:35 -0800278 if (config.HasClientRequestedIndependentOption(kILD0, perspective)) {
fayang3be090d2020-02-11 14:05:08 -0800279 uber_loss_algorithm_.SetReorderingShift(kDefaultIetfLossDelayShift);
280 uber_loss_algorithm_.DisableAdaptiveReorderingThreshold();
fayanga590b752020-01-13 06:37:35 -0800281 }
282 if (config.HasClientRequestedIndependentOption(kILD1, perspective)) {
fayanga590b752020-01-13 06:37:35 -0800283 uber_loss_algorithm_.SetReorderingShift(kDefaultLossDelayShift);
fayang3be090d2020-02-11 14:05:08 -0800284 uber_loss_algorithm_.DisableAdaptiveReorderingThreshold();
fayanga590b752020-01-13 06:37:35 -0800285 }
286 if (config.HasClientRequestedIndependentOption(kILD2, perspective)) {
fayanga590b752020-01-13 06:37:35 -0800287 uber_loss_algorithm_.EnableAdaptiveReorderingThreshold();
fayang3be090d2020-02-11 14:05:08 -0800288 uber_loss_algorithm_.SetReorderingShift(kDefaultIetfLossDelayShift);
fayanga590b752020-01-13 06:37:35 -0800289 }
290 if (config.HasClientRequestedIndependentOption(kILD3, perspective)) {
fayanga590b752020-01-13 06:37:35 -0800291 uber_loss_algorithm_.SetReorderingShift(kDefaultLossDelayShift);
292 uber_loss_algorithm_.EnableAdaptiveReorderingThreshold();
293 }
294 if (config.HasClientRequestedIndependentOption(kILD4, perspective)) {
fayanga590b752020-01-13 06:37:35 -0800295 uber_loss_algorithm_.SetReorderingShift(kDefaultLossDelayShift);
296 uber_loss_algorithm_.EnableAdaptiveReorderingThreshold();
297 uber_loss_algorithm_.EnableAdaptiveTimeThreshold();
fayangb0c7b4b2019-09-12 06:45:24 -0700298 }
fayang84427fc2020-03-09 12:43:26 -0700299 if (GetQuicReloadableFlag(
300 quic_skip_packet_threshold_loss_detection_with_runt) &&
301 config.HasClientRequestedIndependentOption(kRUNT, perspective)) {
302 QUIC_RELOADABLE_FLAG_COUNT_N(
303 quic_skip_packet_threshold_loss_detection_with_runt, 1, 2);
304 uber_loss_algorithm_.DisablePacketThresholdForRuntPackets();
305 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500306 if (config.HasClientSentConnectionOption(kCONH, perspective)) {
307 conservative_handshake_retransmits_ = true;
308 }
309 send_algorithm_->SetFromConfig(config, perspective);
310
311 if (network_change_visitor_ != nullptr) {
312 network_change_visitor_->OnCongestionChange();
313 }
314}
315
316void QuicSentPacketManager::ResumeConnectionState(
317 const CachedNetworkParameters& cached_network_params,
318 bool max_bandwidth_resumption) {
319 QuicBandwidth bandwidth = QuicBandwidth::FromBytesPerSecond(
320 max_bandwidth_resumption
321 ? cached_network_params.max_bandwidth_estimate_bytes_per_second()
322 : cached_network_params.bandwidth_estimate_bytes_per_second());
323 QuicTime::Delta rtt =
324 QuicTime::Delta::FromMilliseconds(cached_network_params.min_rtt_ms());
QUICHE teamb4e187c2019-11-14 06:22:50 -0800325 // This calls the old AdjustNetworkParameters interface, and fills certain
326 // fields in SendAlgorithmInterface::NetworkParams
327 // (e.g., quic_bbr_fix_pacing_rate) using GFE flags.
328 AdjustNetworkParameters(SendAlgorithmInterface::NetworkParams(
329 bandwidth, rtt, /*allow_cwnd_to_decrease = */ false));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500330}
331
fayangf1b99dc2019-05-14 06:29:18 -0700332void QuicSentPacketManager::AdjustNetworkParameters(
QUICHE teamfdcfe3b2019-11-06 10:54:25 -0800333 const SendAlgorithmInterface::NetworkParams& params) {
QUICHE teamb4e187c2019-11-14 06:22:50 -0800334 const QuicBandwidth& bandwidth = params.bandwidth;
335 const QuicTime::Delta& rtt = params.rtt;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500336 if (!rtt.IsZero()) {
337 SetInitialRtt(rtt);
338 }
fayangbe83ecd2019-04-26 13:58:09 -0700339 const QuicByteCount old_cwnd = send_algorithm_->GetCongestionWindow();
fayang6c2d64f2019-05-20 17:12:05 -0700340 if (GetQuicReloadableFlag(quic_conservative_bursts) && using_pacing_ &&
341 !bandwidth.IsZero()) {
fayang4ba55982019-05-13 05:53:22 -0700342 QUIC_RELOADABLE_FLAG_COUNT(quic_conservative_bursts);
343 pacing_sender_.SetBurstTokens(kConservativeUnpacedBurst);
344 }
QUICHE teamb4e187c2019-11-14 06:22:50 -0800345 send_algorithm_->AdjustNetworkParameters(params);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500346 if (debug_delegate_ != nullptr) {
fayangbe83ecd2019-04-26 13:58:09 -0700347 debug_delegate_->OnAdjustNetworkParameters(
fayangef9c8f92019-04-29 13:35:13 -0700348 bandwidth, rtt.IsZero() ? rtt_stats_.SmoothedOrInitialRtt() : rtt,
349 old_cwnd, send_algorithm_->GetCongestionWindow());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500350 }
351}
352
wub44e33052020-03-03 17:11:40 -0800353void QuicSentPacketManager::SetLossDetectionTuner(
354 std::unique_ptr<LossDetectionTunerInterface> tuner) {
355 uber_loss_algorithm_.SetLossDetectionTuner(std::move(tuner));
356}
357
358void QuicSentPacketManager::OnConfigNegotiated() {
359 loss_algorithm_->OnConfigNegotiated();
360}
361
362void QuicSentPacketManager::OnConnectionClosed() {
363 loss_algorithm_->OnConnectionClosed();
364}
365
QUICHE teama6ef0a62019-03-07 20:34:33 -0500366void QuicSentPacketManager::SetHandshakeConfirmed() {
fayang63a19842020-01-23 02:51:28 -0800367 if (!handshake_finished_) {
368 handshake_finished_ = true;
vasilvv975c2352019-10-03 16:53:57 -0400369 NeuterHandshakePackets();
370 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500371}
372
fayang3eb82212019-04-16 12:05:46 -0700373void QuicSentPacketManager::PostProcessNewlyAckedPackets(
fayangf8e918b2019-07-16 13:03:16 -0700374 QuicPacketNumber ack_packet_number,
vasilvv923b6d22020-03-23 11:17:58 -0700375 EncryptionLevel ack_decrypted_level,
QUICHE teama6ef0a62019-03-07 20:34:33 -0500376 const QuicAckFrame& ack_frame,
377 QuicTime ack_receive_time,
378 bool rtt_updated,
379 QuicByteCount prior_bytes_in_flight) {
fayangcff885a2019-10-22 07:39:04 -0700380 unacked_packets_.NotifyAggregatedStreamFrameAcked(
381 last_ack_frame_.ack_delay_time);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500382 InvokeLossDetection(ack_receive_time);
383 // Ignore losses in RTO mode.
384 if (consecutive_rto_count_ > 0 && !use_new_rto_) {
385 packets_lost_.clear();
386 }
387 MaybeInvokeCongestionEvent(rtt_updated, prior_bytes_in_flight,
388 ack_receive_time);
389 unacked_packets_.RemoveObsoletePackets();
390
391 sustained_bandwidth_recorder_.RecordEstimate(
392 send_algorithm_->InRecovery(), send_algorithm_->InSlowStart(),
393 send_algorithm_->BandwidthEstimate(), ack_receive_time, clock_->WallNow(),
394 rtt_stats_.smoothed_rtt());
395
396 // Anytime we are making forward progress and have a new RTT estimate, reset
397 // the backoff counters.
398 if (rtt_updated) {
399 if (consecutive_rto_count_ > 0) {
400 // If the ack acknowledges data sent prior to the RTO,
401 // the RTO was spurious.
402 if (LargestAcked(ack_frame) < first_rto_transmission_) {
403 // Replace SRTT with latest_rtt and increase the variance to prevent
404 // a spurious RTO from happening again.
405 rtt_stats_.ExpireSmoothedMetrics();
406 } else {
407 if (!use_new_rto_) {
408 send_algorithm_->OnRetransmissionTimeout(true);
409 }
410 }
411 }
412 // Reset all retransmit counters any time a new packet is acked.
413 consecutive_rto_count_ = 0;
414 consecutive_tlp_count_ = 0;
fayangce0a3162019-08-15 09:05:36 -0700415 consecutive_pto_count_ = 0;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500416 consecutive_crypto_retransmission_count_ = 0;
417 }
418
419 if (debug_delegate_ != nullptr) {
vasilvv923b6d22020-03-23 11:17:58 -0700420 debug_delegate_->OnIncomingAck(
421 ack_packet_number, ack_decrypted_level, ack_frame, ack_receive_time,
422 LargestAcked(ack_frame), rtt_updated, GetLeastUnacked());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500423 }
424 // Remove packets below least unacked from all_packets_acked_ and
425 // last_ack_frame_.
426 last_ack_frame_.packets.RemoveUpTo(unacked_packets_.GetLeastUnacked());
427 last_ack_frame_.received_packet_times.clear();
428}
429
430void QuicSentPacketManager::MaybeInvokeCongestionEvent(
431 bool rtt_updated,
432 QuicByteCount prior_in_flight,
433 QuicTime event_time) {
434 if (!rtt_updated && packets_acked_.empty() && packets_lost_.empty()) {
435 return;
436 }
437 if (using_pacing_) {
438 pacing_sender_.OnCongestionEvent(rtt_updated, prior_in_flight, event_time,
439 packets_acked_, packets_lost_);
440 } else {
441 send_algorithm_->OnCongestionEvent(rtt_updated, prior_in_flight, event_time,
442 packets_acked_, packets_lost_);
443 }
444 packets_acked_.clear();
445 packets_lost_.clear();
446 if (network_change_visitor_ != nullptr) {
447 network_change_visitor_->OnCongestionChange();
448 }
449}
450
451void QuicSentPacketManager::RetransmitUnackedPackets(
452 TransmissionType retransmission_type) {
453 DCHECK(retransmission_type == ALL_UNACKED_RETRANSMISSION ||
454 retransmission_type == ALL_INITIAL_RETRANSMISSION);
455 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
ianswett898306b2019-04-23 11:05:57 -0700456 for (QuicUnackedPacketMap::iterator it = unacked_packets_.begin();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500457 it != unacked_packets_.end(); ++it, ++packet_number) {
458 if ((retransmission_type == ALL_UNACKED_RETRANSMISSION ||
ianswett898306b2019-04-23 11:05:57 -0700459 it->encryption_level == ENCRYPTION_ZERO_RTT)) {
ianswett04004652019-09-03 18:46:57 -0700460 if (it->in_flight) {
ianswett898306b2019-04-23 11:05:57 -0700461 // Remove 0-RTT packets and packets of the wrong version from flight,
462 // because neither can be processed by the peer.
463 unacked_packets_.RemoveFromInFlight(&*it);
464 }
465 if (unacked_packets_.HasRetransmittableFrames(*it)) {
466 MarkForRetransmission(packet_number, retransmission_type);
467 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500468 }
469 }
ianswett898306b2019-04-23 11:05:57 -0700470 if (retransmission_type == ALL_UNACKED_RETRANSMISSION &&
471 unacked_packets_.bytes_in_flight() > 0) {
472 QUIC_BUG << "RetransmitUnackedPackets should remove all packets from flight"
473 << ", bytes_in_flight:" << unacked_packets_.bytes_in_flight();
474 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500475}
476
477void QuicSentPacketManager::NeuterUnencryptedPackets() {
wubf4ab9652020-02-20 14:45:43 -0800478 for (QuicPacketNumber packet_number :
479 unacked_packets_.NeuterUnencryptedPackets()) {
480 if (avoid_overestimate_bandwidth_with_aggregation_) {
481 QUIC_RELOADABLE_FLAG_COUNT_N(
482 quic_avoid_overestimate_bandwidth_with_aggregation, 1, 4);
483 send_algorithm_->OnPacketNeutered(packet_number);
484 }
485 }
fayang42834942020-03-30 10:56:15 -0700486 if (handshake_mode_disabled_) {
487 consecutive_pto_count_ = 0;
488 uber_loss_algorithm_.ResetLossDetection(INITIAL_DATA);
489 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500490}
491
492void QuicSentPacketManager::NeuterHandshakePackets() {
wubf4ab9652020-02-20 14:45:43 -0800493 for (QuicPacketNumber packet_number :
494 unacked_packets_.NeuterHandshakePackets()) {
495 if (avoid_overestimate_bandwidth_with_aggregation_) {
496 QUIC_RELOADABLE_FLAG_COUNT_N(
497 quic_avoid_overestimate_bandwidth_with_aggregation, 2, 4);
498 send_algorithm_->OnPacketNeutered(packet_number);
499 }
500 }
fayang42834942020-03-30 10:56:15 -0700501 if (handshake_mode_disabled_) {
502 consecutive_pto_count_ = 0;
503 uber_loss_algorithm_.ResetLossDetection(HANDSHAKE_DATA);
504 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500505}
506
fayang4c908f02019-11-01 07:26:17 -0700507bool QuicSentPacketManager::ShouldAddMaxAckDelay() const {
508 DCHECK(pto_enabled_);
509 if (always_include_max_ack_delay_for_pto_timeout_) {
510 return true;
511 }
512 if (!unacked_packets_
513 .GetLargestSentRetransmittableOfPacketNumberSpace(APPLICATION_DATA)
514 .IsInitialized() ||
515 unacked_packets_.GetLargestSentRetransmittableOfPacketNumberSpace(
516 APPLICATION_DATA) <
517 FirstSendingPacketNumber() + kMinReceivedBeforeAckDecimation - 1) {
518 // Peer is doing TCP style acking. Expect an immediate ACK if more than 1
519 // packet are outstanding.
520 if (unacked_packets_.packets_in_flight() >=
521 kDefaultRetransmittablePacketsBeforeAck) {
522 return false;
523 }
524 } else if (unacked_packets_.packets_in_flight() >=
525 kMaxRetransmittablePacketsBeforeAck) {
526 // Peer is doing ack decimation. Expect an immediate ACK if >= 10
527 // packets are outstanding.
528 return false;
529 }
530 if (skip_packet_number_for_pto_ && consecutive_pto_count_ > 0) {
531 // An immediate ACK is expected when doing PTOS. Please note, this will miss
532 // cases when PTO fires and turns out to be spurious.
533 return false;
534 }
535 return true;
536}
537
fayang18ff23b2020-01-28 09:19:00 -0800538QuicTime QuicSentPacketManager::GetEarliestPacketSentTimeForPto(
539 PacketNumberSpace* packet_number_space) const {
540 DCHECK(supports_multiple_packet_number_spaces());
541 QuicTime earliest_sent_time = QuicTime::Zero();
542 for (int8_t i = 0; i < NUM_PACKET_NUMBER_SPACES; ++i) {
543 const QuicTime sent_time = unacked_packets_.GetLastInFlightPacketSentTime(
544 static_cast<PacketNumberSpace>(i));
fayang9a0e1bd2020-02-19 13:13:04 -0800545 if (!ShouldArmPtoForApplicationData() && i == APPLICATION_DATA) {
fayang18ff23b2020-01-28 09:19:00 -0800546 continue;
547 }
fayang9a0e1bd2020-02-19 13:13:04 -0800548 if (!sent_time.IsInitialized() || (earliest_sent_time.IsInitialized() &&
549 earliest_sent_time <= sent_time)) {
550 continue;
fayang18ff23b2020-01-28 09:19:00 -0800551 }
fayang9a0e1bd2020-02-19 13:13:04 -0800552 earliest_sent_time = sent_time;
553 *packet_number_space = static_cast<PacketNumberSpace>(i);
fayang18ff23b2020-01-28 09:19:00 -0800554 }
fayang9a0e1bd2020-02-19 13:13:04 -0800555
fayang18ff23b2020-01-28 09:19:00 -0800556 return earliest_sent_time;
557}
558
fayang9a0e1bd2020-02-19 13:13:04 -0800559bool QuicSentPacketManager::ShouldArmPtoForApplicationData() const {
560 DCHECK(supports_multiple_packet_number_spaces());
561 // Application data must be ignored before handshake completes (1-RTT key
562 // is available). Not arming PTO for application data to prioritize the
563 // completion of handshake. On the server side, handshake_finished_
564 // indicates handshake complete (and confirmed). On the client side,
565 // one_rtt_packet_sent_ indicates handshake complete (while handshake
566 // confirmation will happen later).
567 return handshake_finished_ ||
568 (unacked_packets_.perspective() == Perspective::IS_CLIENT &&
569 one_rtt_packet_sent_);
570}
571
QUICHE teama6ef0a62019-03-07 20:34:33 -0500572void QuicSentPacketManager::MarkForRetransmission(
573 QuicPacketNumber packet_number,
574 TransmissionType transmission_type) {
575 QuicTransmissionInfo* transmission_info =
576 unacked_packets_.GetMutableTransmissionInfo(packet_number);
fayangcff885a2019-10-22 07:39:04 -0700577 // A previous RTO retransmission may cause connection close; packets without
578 // retransmittable frames can be marked for loss retransmissions.
579 QUIC_BUG_IF(transmission_type != LOSS_RETRANSMISSION &&
580 transmission_type != RTO_RETRANSMISSION &&
QUICHE teama6ef0a62019-03-07 20:34:33 -0500581 !unacked_packets_.HasRetransmittableFrames(*transmission_info))
dschinazief79a5f2019-10-04 10:32:54 -0700582 << "transmission_type: " << TransmissionTypeToString(transmission_type);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500583 // Handshake packets should never be sent as probing retransmissions.
fayang73d0ac42019-10-31 12:45:31 -0700584 DCHECK(!transmission_info->has_crypto_handshake ||
QUICHE teama6ef0a62019-03-07 20:34:33 -0500585 transmission_type != PROBING_RETRANSMISSION);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500586
587 HandleRetransmission(transmission_type, transmission_info);
588
589 // Update packet state according to transmission type.
590 transmission_info->state =
591 QuicUtils::RetransmissionTypeToPacketState(transmission_type);
592}
593
594void QuicSentPacketManager::HandleRetransmission(
595 TransmissionType transmission_type,
596 QuicTransmissionInfo* transmission_info) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500597 if (ShouldForceRetransmission(transmission_type)) {
598 // TODO(fayang): Consider to make RTO and PROBING retransmission
599 // strategies be configurable by applications. Today, TLP, RTO and PROBING
600 // retransmissions are handled similarly, i.e., always retranmist the
601 // oldest outstanding data. This is not ideal in general because different
602 // applications may want different strategies. For example, some
603 // applications may want to use higher priority stream data for bandwidth
604 // probing, and some applications want to consider RTO is an indication of
605 // loss, etc.
606 unacked_packets_.RetransmitFrames(*transmission_info, transmission_type);
607 return;
608 }
609
610 unacked_packets_.NotifyFramesLost(*transmission_info, transmission_type);
611 if (transmission_info->retransmittable_frames.empty()) {
612 return;
613 }
614
615 if (transmission_type == LOSS_RETRANSMISSION) {
616 // Record the first packet sent after loss, which allows to wait 1
617 // more RTT before giving up on this lost packet.
618 transmission_info->retransmission =
619 unacked_packets_.largest_sent_packet() + 1;
620 } else {
621 // Clear the recorded first packet sent after loss when version or
622 // encryption changes.
623 transmission_info->retransmission.Clear();
624 }
625}
626
627void QuicSentPacketManager::RecordOneSpuriousRetransmission(
628 const QuicTransmissionInfo& info) {
629 stats_->bytes_spuriously_retransmitted += info.bytes_sent;
630 ++stats_->packets_spuriously_retransmitted;
631 if (debug_delegate_ != nullptr) {
632 debug_delegate_->OnSpuriousPacketRetransmission(info.transmission_type,
633 info.bytes_sent);
634 }
635}
636
QUICHE teama6ef0a62019-03-07 20:34:33 -0500637void QuicSentPacketManager::MarkPacketHandled(QuicPacketNumber packet_number,
638 QuicTransmissionInfo* info,
fayang19d2d5b2019-09-11 14:22:03 -0700639 QuicTime ack_receive_time,
QUICHE team9467db02019-05-30 09:38:45 -0700640 QuicTime::Delta ack_delay_time,
641 QuicTime receive_timestamp) {
fayangcff885a2019-10-22 07:39:04 -0700642 // Try to aggregate acked stream frames if acked packet is not a
643 // retransmission.
644 if (info->transmission_type == NOT_RETRANSMISSION) {
645 unacked_packets_.MaybeAggregateAckedStreamFrame(*info, ack_delay_time,
646 receive_timestamp);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500647 } else {
fayangcff885a2019-10-22 07:39:04 -0700648 unacked_packets_.NotifyAggregatedStreamFrameAcked(ack_delay_time);
649 const bool new_data_acked = unacked_packets_.NotifyFramesAcked(
650 *info, ack_delay_time, receive_timestamp);
651 if (!new_data_acked && info->transmission_type != NOT_RETRANSMISSION) {
652 // Record as a spurious retransmission if this packet is a
653 // retransmission and no new data gets acked.
654 QUIC_DVLOG(1) << "Detect spurious retransmitted packet " << packet_number
655 << " transmission type: "
656 << TransmissionTypeToString(info->transmission_type);
fayangcc4ea6a2019-10-25 08:44:03 -0700657 RecordOneSpuriousRetransmission(*info);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500658 }
659 }
fayangcc4ea6a2019-10-25 08:44:03 -0700660 if (info->state == LOST) {
fayangcff885a2019-10-22 07:39:04 -0700661 // Record as a spurious loss as a packet previously declared lost gets
662 // acked.
fayangcff885a2019-10-22 07:39:04 -0700663 const PacketNumberSpace packet_number_space =
664 unacked_packets_.GetPacketNumberSpace(info->encryption_level);
665 const QuicPacketNumber previous_largest_acked =
666 supports_multiple_packet_number_spaces()
667 ? unacked_packets_.GetLargestAckedOfPacketNumberSpace(
668 packet_number_space)
669 : unacked_packets_.largest_acked();
670 QUIC_DVLOG(1) << "Packet " << packet_number
671 << " was detected lost spuriously, "
672 "previous_largest_acked: "
673 << previous_largest_acked;
674 loss_algorithm_->SpuriousLossDetected(unacked_packets_, rtt_stats_,
675 ack_receive_time, packet_number,
676 previous_largest_acked);
wub02831dc2020-02-28 12:00:54 -0800677 ++stats_->packet_spuriously_detected_lost;
fayangcff885a2019-10-22 07:39:04 -0700678 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500679
680 if (network_change_visitor_ != nullptr &&
681 info->bytes_sent > largest_mtu_acked_) {
682 largest_mtu_acked_ = info->bytes_sent;
683 network_change_visitor_->OnPathMtuIncreased(largest_mtu_acked_);
684 }
685 unacked_packets_.RemoveFromInFlight(info);
686 unacked_packets_.RemoveRetransmittability(info);
687 info->state = ACKED;
688}
689
690bool QuicSentPacketManager::OnPacketSent(
691 SerializedPacket* serialized_packet,
QUICHE teama6ef0a62019-03-07 20:34:33 -0500692 QuicTime sent_time,
693 TransmissionType transmission_type,
694 HasRetransmittableData has_retransmittable_data) {
695 QuicPacketNumber packet_number = serialized_packet->packet_number;
696 DCHECK_LE(FirstSendingPacketNumber(), packet_number);
697 DCHECK(!unacked_packets_.IsUnacked(packet_number));
698 QUIC_BUG_IF(serialized_packet->encrypted_length == 0)
699 << "Cannot send empty packets.";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500700 if (pending_timer_transmission_count_ > 0) {
701 --pending_timer_transmission_count_;
702 }
703
704 bool in_flight = has_retransmittable_data == HAS_RETRANSMITTABLE_DATA;
705 if (using_pacing_) {
706 pacing_sender_.OnPacketSent(
707 sent_time, unacked_packets_.bytes_in_flight(), packet_number,
708 serialized_packet->encrypted_length, has_retransmittable_data);
709 } else {
710 send_algorithm_->OnPacketSent(
711 sent_time, unacked_packets_.bytes_in_flight(), packet_number,
712 serialized_packet->encrypted_length, has_retransmittable_data);
713 }
714
fayang9a0e1bd2020-02-19 13:13:04 -0800715 if (serialized_packet->encryption_level == ENCRYPTION_FORWARD_SECURE) {
716 one_rtt_packet_sent_ = true;
717 }
718
fayangcff885a2019-10-22 07:39:04 -0700719 unacked_packets_.AddSentPacket(serialized_packet, transmission_type,
720 sent_time, in_flight);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500721 // Reset the retransmission timer anytime a pending packet is sent.
722 return in_flight;
723}
724
fayang67f82272019-08-14 16:08:45 -0700725QuicSentPacketManager::RetransmissionTimeoutMode
726QuicSentPacketManager::OnRetransmissionTimeout() {
fayang5f135052019-08-22 17:59:40 -0700727 DCHECK(unacked_packets_.HasInFlightPackets() ||
fayang63a19842020-01-23 02:51:28 -0800728 (handshake_mode_disabled_ && !handshake_finished_));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500729 DCHECK_EQ(0u, pending_timer_transmission_count_);
730 // Handshake retransmission, timer based loss detection, TLP, and RTO are
731 // implemented with a single alarm. The handshake alarm is set when the
732 // handshake has not completed, the loss alarm is set when the loss detection
733 // algorithm says to, and the TLP and RTO alarms are set after that.
734 // The TLP alarm is always set to run for under an RTO.
735 switch (GetRetransmissionMode()) {
736 case HANDSHAKE_MODE:
fayang5f135052019-08-22 17:59:40 -0700737 DCHECK(!handshake_mode_disabled_);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500738 ++stats_->crypto_retransmit_count;
739 RetransmitCryptoPackets();
fayang67f82272019-08-14 16:08:45 -0700740 return HANDSHAKE_MODE;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500741 case LOSS_MODE: {
742 ++stats_->loss_timeout_count;
743 QuicByteCount prior_in_flight = unacked_packets_.bytes_in_flight();
744 const QuicTime now = clock_->Now();
745 InvokeLossDetection(now);
746 MaybeInvokeCongestionEvent(false, prior_in_flight, now);
fayang67f82272019-08-14 16:08:45 -0700747 return LOSS_MODE;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500748 }
749 case TLP_MODE:
750 ++stats_->tlp_count;
751 ++consecutive_tlp_count_;
752 pending_timer_transmission_count_ = 1;
753 // TLPs prefer sending new data instead of retransmitting data, so
754 // give the connection a chance to write before completing the TLP.
fayang67f82272019-08-14 16:08:45 -0700755 return TLP_MODE;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500756 case RTO_MODE:
757 ++stats_->rto_count;
758 RetransmitRtoPackets();
fayang67f82272019-08-14 16:08:45 -0700759 return RTO_MODE;
fayangce0a3162019-08-15 09:05:36 -0700760 case PTO_MODE:
761 QUIC_DVLOG(1) << ENDPOINT << "PTO mode";
762 ++stats_->pto_count;
763 ++consecutive_pto_count_;
764 pending_timer_transmission_count_ = max_probe_packets_per_pto_;
765 return PTO_MODE;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500766 }
767}
768
769void QuicSentPacketManager::RetransmitCryptoPackets() {
770 DCHECK_EQ(HANDSHAKE_MODE, GetRetransmissionMode());
771 ++consecutive_crypto_retransmission_count_;
772 bool packet_retransmitted = false;
773 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
774 std::vector<QuicPacketNumber> crypto_retransmissions;
775 for (QuicUnackedPacketMap::const_iterator it = unacked_packets_.begin();
776 it != unacked_packets_.end(); ++it, ++packet_number) {
777 // Only retransmit frames which are in flight, and therefore have been sent.
fayangcff885a2019-10-22 07:39:04 -0700778 if (!it->in_flight || it->state != OUTSTANDING ||
QUICHE teama6ef0a62019-03-07 20:34:33 -0500779 !it->has_crypto_handshake ||
780 !unacked_packets_.HasRetransmittableFrames(*it)) {
781 continue;
782 }
783 packet_retransmitted = true;
fayangcff885a2019-10-22 07:39:04 -0700784 crypto_retransmissions.push_back(packet_number);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500785 ++pending_timer_transmission_count_;
786 }
787 DCHECK(packet_retransmitted) << "No crypto packets found to retransmit.";
fayangcff885a2019-10-22 07:39:04 -0700788 for (QuicPacketNumber retransmission : crypto_retransmissions) {
789 MarkForRetransmission(retransmission, HANDSHAKE_RETRANSMISSION);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500790 }
791}
792
793bool QuicSentPacketManager::MaybeRetransmitTailLossProbe() {
fayang62f867a2019-08-22 12:05:01 -0700794 DCHECK(!pto_enabled_);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500795 if (pending_timer_transmission_count_ == 0) {
796 return false;
797 }
798 if (!MaybeRetransmitOldestPacket(TLP_RETRANSMISSION)) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500799 return false;
800 }
801 return true;
802}
803
804bool QuicSentPacketManager::MaybeRetransmitOldestPacket(TransmissionType type) {
805 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
806 for (QuicUnackedPacketMap::const_iterator it = unacked_packets_.begin();
807 it != unacked_packets_.end(); ++it, ++packet_number) {
808 // Only retransmit frames which are in flight, and therefore have been sent.
fayangcff885a2019-10-22 07:39:04 -0700809 if (!it->in_flight || it->state != OUTSTANDING ||
QUICHE teama6ef0a62019-03-07 20:34:33 -0500810 !unacked_packets_.HasRetransmittableFrames(*it)) {
811 continue;
812 }
813 MarkForRetransmission(packet_number, type);
814 return true;
815 }
816 QUIC_DVLOG(1)
817 << "No retransmittable packets, so RetransmitOldestPacket failed.";
818 return false;
819}
820
821void QuicSentPacketManager::RetransmitRtoPackets() {
fayang62f867a2019-08-22 12:05:01 -0700822 DCHECK(!pto_enabled_);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500823 QUIC_BUG_IF(pending_timer_transmission_count_ > 0)
824 << "Retransmissions already queued:" << pending_timer_transmission_count_;
825 // Mark two packets for retransmission.
826 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
827 std::vector<QuicPacketNumber> retransmissions;
828 for (QuicUnackedPacketMap::const_iterator it = unacked_packets_.begin();
829 it != unacked_packets_.end(); ++it, ++packet_number) {
fayangcff885a2019-10-22 07:39:04 -0700830 if (it->state == OUTSTANDING &&
QUICHE teama6ef0a62019-03-07 20:34:33 -0500831 unacked_packets_.HasRetransmittableFrames(*it) &&
832 pending_timer_transmission_count_ < max_rto_packets_) {
fayange1e81d22020-01-06 11:41:34 -0800833 DCHECK(it->in_flight);
fayangcff885a2019-10-22 07:39:04 -0700834 retransmissions.push_back(packet_number);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500835 ++pending_timer_transmission_count_;
836 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500837 }
838 if (pending_timer_transmission_count_ > 0) {
839 if (consecutive_rto_count_ == 0) {
840 first_rto_transmission_ = unacked_packets_.largest_sent_packet() + 1;
841 }
842 ++consecutive_rto_count_;
843 }
fayangcff885a2019-10-22 07:39:04 -0700844 for (QuicPacketNumber retransmission : retransmissions) {
845 MarkForRetransmission(retransmission, RTO_RETRANSMISSION);
846 }
847 if (retransmissions.empty()) {
848 QUIC_BUG_IF(pending_timer_transmission_count_ != 0);
849 // No packets to be RTO retransmitted, raise up a credit to allow
850 // connection to send.
851 QUIC_CODE_COUNT(no_packets_to_be_rto_retransmitted);
852 pending_timer_transmission_count_ = 1;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500853 }
854}
855
fayangce0a3162019-08-15 09:05:36 -0700856void QuicSentPacketManager::MaybeSendProbePackets() {
857 if (pending_timer_transmission_count_ == 0) {
858 return;
859 }
fayang18ff23b2020-01-28 09:19:00 -0800860 PacketNumberSpace packet_number_space;
861 if (supports_multiple_packet_number_spaces()) {
862 // Find out the packet number space to send probe packets.
863 if (!GetEarliestPacketSentTimeForPto(&packet_number_space)
864 .IsInitialized()) {
865 QUIC_BUG << "earlist_sent_time not initialized when trying to send PTO "
866 "retransmissions";
867 return;
868 }
869 }
fayangce0a3162019-08-15 09:05:36 -0700870 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
871 std::vector<QuicPacketNumber> probing_packets;
872 for (QuicUnackedPacketMap::const_iterator it = unacked_packets_.begin();
873 it != unacked_packets_.end(); ++it, ++packet_number) {
874 if (it->state == OUTSTANDING &&
fayang18ff23b2020-01-28 09:19:00 -0800875 unacked_packets_.HasRetransmittableFrames(*it) &&
876 (!supports_multiple_packet_number_spaces() ||
877 unacked_packets_.GetPacketNumberSpace(it->encryption_level) ==
878 packet_number_space)) {
fayange1e81d22020-01-06 11:41:34 -0800879 DCHECK(it->in_flight);
fayangce0a3162019-08-15 09:05:36 -0700880 probing_packets.push_back(packet_number);
881 if (probing_packets.size() == pending_timer_transmission_count_) {
882 break;
883 }
884 }
885 }
886
887 for (QuicPacketNumber retransmission : probing_packets) {
888 QUIC_DVLOG(1) << ENDPOINT << "Marking " << retransmission
889 << " for probing retransmission";
fayang73d0ac42019-10-31 12:45:31 -0700890 MarkForRetransmission(retransmission, PTO_RETRANSMISSION);
fayangce0a3162019-08-15 09:05:36 -0700891 }
892 // It is possible that there is not enough outstanding data for probing.
893}
894
895void QuicSentPacketManager::AdjustPendingTimerTransmissions() {
896 if (pending_timer_transmission_count_ < max_probe_packets_per_pto_) {
897 // There are packets sent already, clear credit.
898 pending_timer_transmission_count_ = 0;
899 return;
900 }
901 // No packet gets sent, leave 1 credit to allow data to be write eventually.
902 pending_timer_transmission_count_ = 1;
903}
904
fayang08a6c952019-09-18 14:29:45 -0700905void QuicSentPacketManager::EnableIetfPtoAndLossDetection() {
fayang5f135052019-08-22 17:59:40 -0700906 pto_enabled_ = true;
907 handshake_mode_disabled_ = true;
fayang1c2d1ab2020-03-11 12:08:41 -0700908 // Default to 1 packet per PTO and skip a packet number. Arm the 1st PTO with
909 // max of earliest in flight sent time + PTO delay and 1.5 * srtt from
910 // last in flight packet.
911 max_probe_packets_per_pto_ = 1;
912 skip_packet_number_for_pto_ = true;
913 first_pto_srtt_multiplier_ = 1.5;
fayang5f135052019-08-22 17:59:40 -0700914}
915
fayang7085a6d2019-11-04 07:03:57 -0800916void QuicSentPacketManager::StartExponentialBackoffAfterNthPto(
917 size_t exponential_backoff_start_point) {
918 pto_exponential_backoff_start_point_ = exponential_backoff_start_point;
919}
920
QUICHE teama6ef0a62019-03-07 20:34:33 -0500921QuicSentPacketManager::RetransmissionTimeoutMode
922QuicSentPacketManager::GetRetransmissionMode() const {
fayang5f135052019-08-22 17:59:40 -0700923 DCHECK(unacked_packets_.HasInFlightPackets() ||
fayang63a19842020-01-23 02:51:28 -0800924 (handshake_mode_disabled_ && !handshake_finished_));
925 if (!handshake_mode_disabled_ && !handshake_finished_ &&
fayang5f135052019-08-22 17:59:40 -0700926 unacked_packets_.HasPendingCryptoPackets()) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500927 return HANDSHAKE_MODE;
928 }
929 if (loss_algorithm_->GetLossTimeout() != QuicTime::Zero()) {
930 return LOSS_MODE;
931 }
fayang62f867a2019-08-22 12:05:01 -0700932 if (pto_enabled_) {
fayangce0a3162019-08-15 09:05:36 -0700933 return PTO_MODE;
934 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500935 if (consecutive_tlp_count_ < max_tail_loss_probes_) {
ianswett7ddb34f2019-07-15 12:17:43 -0700936 if (unacked_packets_.HasUnackedRetransmittableFrames()) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500937 return TLP_MODE;
938 }
939 }
940 return RTO_MODE;
941}
942
943void QuicSentPacketManager::InvokeLossDetection(QuicTime time) {
944 if (!packets_acked_.empty()) {
945 DCHECK_LE(packets_acked_.front().packet_number,
946 packets_acked_.back().packet_number);
947 largest_newly_acked_ = packets_acked_.back().packet_number;
948 }
949 loss_algorithm_->DetectLosses(unacked_packets_, time, rtt_stats_,
950 largest_newly_acked_, packets_acked_,
951 &packets_lost_);
952 for (const LostPacket& packet : packets_lost_) {
wube330aa42020-02-27 07:49:09 -0800953 QuicTransmissionInfo* info =
954 unacked_packets_.GetMutableTransmissionInfo(packet.packet_number);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500955 ++stats_->packets_lost;
wube330aa42020-02-27 07:49:09 -0800956 if (time > info->sent_time) {
957 stats_->total_loss_detection_time =
958 stats_->total_loss_detection_time + (time - info->sent_time);
959 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500960 if (debug_delegate_ != nullptr) {
vasilvv923b6d22020-03-23 11:17:58 -0700961 debug_delegate_->OnPacketLoss(packet.packet_number,
962 info->encryption_level, LOSS_RETRANSMISSION,
QUICHE teama6ef0a62019-03-07 20:34:33 -0500963 time);
964 }
wube330aa42020-02-27 07:49:09 -0800965 unacked_packets_.RemoveFromInFlight(info);
966
QUICHE teama6ef0a62019-03-07 20:34:33 -0500967 MarkForRetransmission(packet.packet_number, LOSS_RETRANSMISSION);
968 }
969}
970
971bool QuicSentPacketManager::MaybeUpdateRTT(QuicPacketNumber largest_acked,
972 QuicTime::Delta ack_delay_time,
973 QuicTime ack_receive_time) {
974 // We rely on ack_delay_time to compute an RTT estimate, so we
975 // only update rtt when the largest observed gets acked.
976 if (!unacked_packets_.IsUnacked(largest_acked)) {
977 return false;
978 }
979 // We calculate the RTT based on the highest ACKed packet number, the lower
980 // packet numbers will include the ACK aggregation delay.
981 const QuicTransmissionInfo& transmission_info =
982 unacked_packets_.GetTransmissionInfo(largest_acked);
983 // Ensure the packet has a valid sent time.
984 if (transmission_info.sent_time == QuicTime::Zero()) {
985 QUIC_BUG << "Acked packet has zero sent time, largest_acked:"
986 << largest_acked;
987 return false;
988 }
989 if (transmission_info.sent_time > ack_receive_time) {
990 QUIC_CODE_COUNT(quic_receive_acked_before_sending);
991 }
992
993 QuicTime::Delta send_delta = ack_receive_time - transmission_info.sent_time;
wub44e33052020-03-03 17:11:40 -0800994 const bool min_rtt_available = !rtt_stats_.min_rtt().IsZero();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500995 rtt_stats_.UpdateRtt(send_delta, ack_delay_time, ack_receive_time);
996
wub44e33052020-03-03 17:11:40 -0800997 if (!min_rtt_available && !rtt_stats_.min_rtt().IsZero()) {
998 loss_algorithm_->OnMinRttAvailable();
999 }
1000
QUICHE teama6ef0a62019-03-07 20:34:33 -05001001 return true;
1002}
1003
1004QuicTime::Delta QuicSentPacketManager::TimeUntilSend(QuicTime now) const {
1005 // The TLP logic is entirely contained within QuicSentPacketManager, so the
1006 // send algorithm does not need to be consulted.
1007 if (pending_timer_transmission_count_ > 0) {
1008 return QuicTime::Delta::Zero();
1009 }
1010
1011 if (using_pacing_) {
1012 return pacing_sender_.TimeUntilSend(now,
1013 unacked_packets_.bytes_in_flight());
1014 }
1015
1016 return send_algorithm_->CanSend(unacked_packets_.bytes_in_flight())
1017 ? QuicTime::Delta::Zero()
1018 : QuicTime::Delta::Infinite();
1019}
1020
1021const QuicTime QuicSentPacketManager::GetRetransmissionTime() const {
fayang5f135052019-08-22 17:59:40 -07001022 if (!unacked_packets_.HasInFlightPackets() &&
fayang63a19842020-01-23 02:51:28 -08001023 (!handshake_mode_disabled_ || handshake_finished_ ||
fayang5f135052019-08-22 17:59:40 -07001024 unacked_packets_.perspective() == Perspective::IS_SERVER)) {
1025 // Do not set the timer if there is nothing in flight. However, to avoid
1026 // handshake deadlock due to anti-amplification limit, client needs to set
1027 // PTO timer when the handshake is not confirmed even there is nothing in
1028 // flight.
1029 return QuicTime::Zero();
1030 }
1031 if (pending_timer_transmission_count_ > 0) {
1032 // Do not set the timer if there is any credit left.
QUICHE teama6ef0a62019-03-07 20:34:33 -05001033 return QuicTime::Zero();
1034 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001035 switch (GetRetransmissionMode()) {
1036 case HANDSHAKE_MODE:
1037 return unacked_packets_.GetLastCryptoPacketSentTime() +
1038 GetCryptoRetransmissionDelay();
1039 case LOSS_MODE:
1040 return loss_algorithm_->GetLossTimeout();
1041 case TLP_MODE: {
fayang62f867a2019-08-22 12:05:01 -07001042 DCHECK(!pto_enabled_);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001043 // TODO(ianswett): When CWND is available, it would be preferable to
1044 // set the timer based on the earliest retransmittable packet.
1045 // Base the updated timer on the send time of the last packet.
ianswett479fea32019-09-04 02:51:01 -07001046 const QuicTime sent_time =
1047 unacked_packets_.GetLastInFlightPacketSentTime();
QUICHE teama6ef0a62019-03-07 20:34:33 -05001048 const QuicTime tlp_time = sent_time + GetTailLossProbeDelay();
1049 // Ensure the TLP timer never gets set to a time in the past.
1050 return std::max(clock_->ApproximateNow(), tlp_time);
1051 }
1052 case RTO_MODE: {
fayang62f867a2019-08-22 12:05:01 -07001053 DCHECK(!pto_enabled_);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001054 // The RTO is based on the first outstanding packet.
ianswett479fea32019-09-04 02:51:01 -07001055 const QuicTime sent_time =
1056 unacked_packets_.GetLastInFlightPacketSentTime();
QUICHE teama6ef0a62019-03-07 20:34:33 -05001057 QuicTime rto_time = sent_time + GetRetransmissionDelay();
1058 // Wait for TLP packets to be acked before an RTO fires.
ianswett479fea32019-09-04 02:51:01 -07001059 QuicTime tlp_time = sent_time + GetTailLossProbeDelay();
QUICHE teama6ef0a62019-03-07 20:34:33 -05001060 return std::max(tlp_time, rto_time);
1061 }
fayangce0a3162019-08-15 09:05:36 -07001062 case PTO_MODE: {
fayang18ff23b2020-01-28 09:19:00 -08001063 if (!supports_multiple_packet_number_spaces()) {
fayangc5d9b2c2020-03-02 11:12:26 -08001064 if (first_pto_srtt_multiplier_ > 0 &&
fayang2ccfbcf2020-02-28 12:37:08 -08001065 unacked_packets_.HasInFlightPackets() &&
1066 consecutive_pto_count_ == 0) {
1067 // Arm 1st PTO with earliest in flight sent time, and make sure at
fayangc5d9b2c2020-03-02 11:12:26 -08001068 // least first_pto_srtt_multiplier_ * RTT has been passed since last
1069 // in flight packet.
fayang2ccfbcf2020-02-28 12:37:08 -08001070 return std::max(
1071 clock_->ApproximateNow(),
1072 std::max(unacked_packets_.GetFirstInFlightTransmissionInfo()
1073 ->sent_time +
1074 GetProbeTimeoutDelay(),
1075 unacked_packets_.GetLastInFlightPacketSentTime() +
fayangc5d9b2c2020-03-02 11:12:26 -08001076 first_pto_srtt_multiplier_ *
1077 rtt_stats_.SmoothedOrInitialRtt()));
fayang2ccfbcf2020-02-28 12:37:08 -08001078 }
fayang18ff23b2020-01-28 09:19:00 -08001079 // Ensure PTO never gets set to a time in the past.
1080 return std::max(clock_->ApproximateNow(),
1081 unacked_packets_.GetLastInFlightPacketSentTime() +
1082 GetProbeTimeoutDelay());
1083 }
1084
fayang3ecc53b2020-03-12 07:12:03 -07001085 PacketNumberSpace packet_number_space = NUM_PACKET_NUMBER_SPACES;
fayang2ccfbcf2020-02-28 12:37:08 -08001086 // earliest_right_edge is the earliest sent time of the last in flight
1087 // packet of all packet number spaces.
1088 const QuicTime earliest_right_edge =
1089 GetEarliestPacketSentTimeForPto(&packet_number_space);
fayangc5d9b2c2020-03-02 11:12:26 -08001090 if (first_pto_srtt_multiplier_ > 0 &&
fayang2ccfbcf2020-02-28 12:37:08 -08001091 packet_number_space == APPLICATION_DATA &&
1092 consecutive_pto_count_ == 0) {
1093 const QuicTransmissionInfo* first_application_info =
1094 unacked_packets_.GetFirstInFlightTransmissionInfoOfSpace(
1095 APPLICATION_DATA);
1096 if (first_application_info != nullptr) {
1097 // Arm 1st PTO with earliest in flight sent time, and make sure at
fayangc5d9b2c2020-03-02 11:12:26 -08001098 // least first_pto_srtt_multiplier_ * RTT has been passed since last
1099 // in flight packet. Only do this for application data.
fayang2ccfbcf2020-02-28 12:37:08 -08001100 return std::max(
1101 clock_->ApproximateNow(),
1102 std::max(
1103 first_application_info->sent_time + GetProbeTimeoutDelay(),
fayangc5d9b2c2020-03-02 11:12:26 -08001104 earliest_right_edge + first_pto_srtt_multiplier_ *
1105 rtt_stats_.SmoothedOrInitialRtt()));
fayang2ccfbcf2020-02-28 12:37:08 -08001106 }
1107 }
ianswett479fea32019-09-04 02:51:01 -07001108 return std::max(clock_->ApproximateNow(),
fayang2ccfbcf2020-02-28 12:37:08 -08001109 earliest_right_edge + GetProbeTimeoutDelay());
fayangce0a3162019-08-15 09:05:36 -07001110 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001111 }
1112 DCHECK(false);
1113 return QuicTime::Zero();
1114}
1115
1116const QuicTime::Delta QuicSentPacketManager::GetPathDegradingDelay() const {
fayang1640cc22020-03-21 14:11:02 -07001117 return GetNConsecutiveRetransmissionTimeoutDelay(
1118 max_tail_loss_probes_ + kNumRetransmissionDelaysForPathDegradingDelay);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001119}
1120
fayangb59c6f12020-03-23 15:06:14 -07001121const QuicTime::Delta QuicSentPacketManager::GetNetworkBlackholeDelay() const {
1122 return GetNConsecutiveRetransmissionTimeoutDelay(
1123 max_tail_loss_probes_ + kNumRetransmissionDelaysForBlackholeDelay);
1124}
1125
QUICHE teama6ef0a62019-03-07 20:34:33 -05001126const QuicTime::Delta QuicSentPacketManager::GetCryptoRetransmissionDelay()
1127 const {
1128 // This is equivalent to the TailLossProbeDelay, but slightly more aggressive
1129 // because crypto handshake messages don't incur a delayed ack time.
1130 QuicTime::Delta srtt = rtt_stats_.SmoothedOrInitialRtt();
1131 int64_t delay_ms;
1132 if (conservative_handshake_retransmits_) {
1133 // Using the delayed ack time directly could cause conservative handshake
1134 // retransmissions to actually be more aggressive than the default.
fkastenholz59c653b2019-07-15 09:55:53 -07001135 delay_ms = std::max(peer_max_ack_delay_.ToMilliseconds(),
QUICHE teama6ef0a62019-03-07 20:34:33 -05001136 static_cast<int64_t>(2 * srtt.ToMilliseconds()));
1137 } else {
1138 delay_ms = std::max(kMinHandshakeTimeoutMs,
1139 static_cast<int64_t>(1.5 * srtt.ToMilliseconds()));
1140 }
1141 return QuicTime::Delta::FromMilliseconds(
1142 delay_ms << consecutive_crypto_retransmission_count_);
1143}
1144
fayang1640cc22020-03-21 14:11:02 -07001145const QuicTime::Delta QuicSentPacketManager::GetTailLossProbeDelay() const {
QUICHE teama6ef0a62019-03-07 20:34:33 -05001146 QuicTime::Delta srtt = rtt_stats_.SmoothedOrInitialRtt();
fayang1640cc22020-03-21 14:11:02 -07001147 if (enable_half_rtt_tail_loss_probe_ && consecutive_tlp_count_ == 0u) {
zhongyi1b2f7832019-06-14 13:31:34 -07001148 if (unacked_packets().HasUnackedStreamData()) {
1149 // Enable TLPR if there are pending data packets.
1150 return std::max(min_tlp_timeout_, srtt * 0.5);
1151 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001152 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001153 if (!unacked_packets_.HasMultipleInFlightPackets()) {
1154 // This expression really should be using the delayed ack time, but in TCP
1155 // MinRTO was traditionally set to 2x the delayed ack timer and this
1156 // expression assumed QUIC did the same.
1157 return std::max(2 * srtt, 1.5 * srtt + (min_rto_timeout_ * 0.5));
1158 }
1159 return std::max(min_tlp_timeout_, 2 * srtt);
1160}
1161
fayang1640cc22020-03-21 14:11:02 -07001162const QuicTime::Delta QuicSentPacketManager::GetRetransmissionDelay() const {
QUICHE teama6ef0a62019-03-07 20:34:33 -05001163 QuicTime::Delta retransmission_delay = QuicTime::Delta::Zero();
1164 if (rtt_stats_.smoothed_rtt().IsZero()) {
1165 // We are in the initial state, use default timeout values.
1166 retransmission_delay =
1167 QuicTime::Delta::FromMilliseconds(kDefaultRetransmissionTimeMs);
1168 } else {
1169 retransmission_delay =
1170 rtt_stats_.smoothed_rtt() + 4 * rtt_stats_.mean_deviation();
1171 if (retransmission_delay < min_rto_timeout_) {
1172 retransmission_delay = min_rto_timeout_;
1173 }
1174 }
1175
1176 // Calculate exponential back off.
1177 retransmission_delay =
1178 retransmission_delay *
fayang1640cc22020-03-21 14:11:02 -07001179 (1 << std::min<size_t>(consecutive_rto_count_, kMaxRetransmissions));
QUICHE teama6ef0a62019-03-07 20:34:33 -05001180
1181 if (retransmission_delay.ToMilliseconds() > kMaxRetransmissionTimeMs) {
1182 return QuicTime::Delta::FromMilliseconds(kMaxRetransmissionTimeMs);
1183 }
1184 return retransmission_delay;
1185}
1186
fayangce0a3162019-08-15 09:05:36 -07001187const QuicTime::Delta QuicSentPacketManager::GetProbeTimeoutDelay() const {
fayang62f867a2019-08-22 12:05:01 -07001188 DCHECK(pto_enabled_);
fayangce0a3162019-08-15 09:05:36 -07001189 if (rtt_stats_.smoothed_rtt().IsZero()) {
1190 if (rtt_stats_.initial_rtt().IsZero()) {
1191 return QuicTime::Delta::FromSeconds(1);
1192 }
1193 return 2 * rtt_stats_.initial_rtt();
1194 }
fayangc2fd6102020-03-02 17:27:38 -08001195 const QuicTime::Delta rtt_var = use_standard_deviation_for_pto_
1196 ? rtt_stats_.GetStandardOrMeanDeviation()
1197 : rtt_stats_.mean_deviation();
fayang4aa22402020-01-07 11:36:07 -08001198 QuicTime::Delta pto_delay =
fayangce0a3162019-08-15 09:05:36 -07001199 rtt_stats_.smoothed_rtt() +
fayangc2fd6102020-03-02 17:27:38 -08001200 std::max(pto_rttvar_multiplier_ * rtt_var, kAlarmGranularity) +
fayang4c908f02019-11-01 07:26:17 -07001201 (ShouldAddMaxAckDelay() ? peer_max_ack_delay_ : QuicTime::Delta::Zero());
fayang4aa22402020-01-07 11:36:07 -08001202 pto_delay =
1203 pto_delay * (1 << (consecutive_pto_count_ -
1204 std::min(consecutive_pto_count_,
1205 pto_exponential_backoff_start_point_)));
1206 if (consecutive_pto_count_ < num_tlp_timeout_ptos_) {
1207 // Make first n PTOs similar to TLPs.
1208 if (pto_delay > 2 * rtt_stats_.smoothed_rtt()) {
1209 QUIC_CODE_COUNT(quic_delayed_pto);
1210 pto_delay = std::max(kAlarmGranularity, 2 * rtt_stats_.smoothed_rtt());
1211 } else {
1212 QUIC_CODE_COUNT(quic_faster_pto);
1213 }
1214 }
1215 return pto_delay;
fayangce0a3162019-08-15 09:05:36 -07001216}
1217
wub967ba572019-04-01 09:27:52 -07001218QuicTime::Delta QuicSentPacketManager::GetSlowStartDuration() const {
wubbe29b9e2019-11-24 06:56:04 -08001219 if (send_algorithm_->GetCongestionControlType() == kBBR ||
1220 send_algorithm_->GetCongestionControlType() == kBBRv2) {
1221 return stats_->slowstart_duration.GetTotalElapsedTime(
1222 clock_->ApproximateNow());
wub967ba572019-04-01 09:27:52 -07001223 }
wubbe29b9e2019-11-24 06:56:04 -08001224 return QuicTime::Delta::Infinite();
wub967ba572019-04-01 09:27:52 -07001225}
1226
vasilvvc48c8712019-03-11 13:38:16 -07001227std::string QuicSentPacketManager::GetDebugState() const {
QUICHE teama6ef0a62019-03-07 20:34:33 -05001228 return send_algorithm_->GetDebugState();
1229}
1230
QUICHE teama6ef0a62019-03-07 20:34:33 -05001231void QuicSentPacketManager::SetSendAlgorithm(
1232 CongestionControlType congestion_control_type) {
wub32a9f8a2020-03-16 09:01:35 -07001233 if (send_algorithm_ &&
1234 send_algorithm_->GetCongestionControlType() == congestion_control_type) {
1235 return;
wub18773d62020-01-28 14:14:06 -08001236 }
wub32a9f8a2020-03-16 09:01:35 -07001237
QUICHE teama6ef0a62019-03-07 20:34:33 -05001238 SetSendAlgorithm(SendAlgorithmInterface::Create(
QUICHE team73957f12019-04-18 16:21:52 -07001239 clock_, &rtt_stats_, &unacked_packets_, congestion_control_type, random_,
1240 stats_, initial_congestion_window_));
QUICHE teama6ef0a62019-03-07 20:34:33 -05001241}
1242
1243void QuicSentPacketManager::SetSendAlgorithm(
1244 SendAlgorithmInterface* send_algorithm) {
1245 send_algorithm_.reset(send_algorithm);
1246 pacing_sender_.set_sender(send_algorithm);
1247}
1248
1249void QuicSentPacketManager::OnConnectionMigration(AddressChangeType type) {
1250 if (type == PORT_CHANGE || type == IPV4_SUBNET_CHANGE) {
1251 // Rtt and cwnd do not need to be reset when the peer address change is
1252 // considered to be caused by NATs.
1253 return;
1254 }
1255 consecutive_rto_count_ = 0;
1256 consecutive_tlp_count_ = 0;
fayangce0a3162019-08-15 09:05:36 -07001257 consecutive_pto_count_ = 0;
QUICHE teama6ef0a62019-03-07 20:34:33 -05001258 rtt_stats_.OnConnectionMigration();
1259 send_algorithm_->OnConnectionMigration();
1260}
1261
1262void QuicSentPacketManager::OnAckFrameStart(QuicPacketNumber largest_acked,
1263 QuicTime::Delta ack_delay_time,
1264 QuicTime ack_receive_time) {
1265 DCHECK(packets_acked_.empty());
1266 DCHECK_LE(largest_acked, unacked_packets_.largest_sent_packet());
wub81c9b662020-02-14 09:08:07 -08001267 if (ack_delay_time > peer_max_ack_delay()) {
wub8ee29a02019-12-12 13:06:48 -08001268 ack_delay_time = peer_max_ack_delay();
1269 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001270 rtt_updated_ =
1271 MaybeUpdateRTT(largest_acked, ack_delay_time, ack_receive_time);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001272 last_ack_frame_.ack_delay_time = ack_delay_time;
1273 acked_packets_iter_ = last_ack_frame_.packets.rbegin();
1274}
1275
1276void QuicSentPacketManager::OnAckRange(QuicPacketNumber start,
1277 QuicPacketNumber end) {
1278 if (!last_ack_frame_.largest_acked.IsInitialized() ||
1279 end > last_ack_frame_.largest_acked + 1) {
1280 // Largest acked increases.
1281 unacked_packets_.IncreaseLargestAcked(end - 1);
1282 last_ack_frame_.largest_acked = end - 1;
1283 }
1284 // Drop ack ranges which ack packets below least_unacked.
1285 QuicPacketNumber least_unacked = unacked_packets_.GetLeastUnacked();
1286 if (least_unacked.IsInitialized() && end <= least_unacked) {
1287 return;
1288 }
1289 start = std::max(start, least_unacked);
1290 do {
1291 QuicPacketNumber newly_acked_start = start;
1292 if (acked_packets_iter_ != last_ack_frame_.packets.rend()) {
1293 newly_acked_start = std::max(start, acked_packets_iter_->max());
1294 }
1295 for (QuicPacketNumber acked = end - 1; acked >= newly_acked_start;
1296 --acked) {
1297 // Check if end is above the current range. If so add newly acked packets
1298 // in descending order.
1299 packets_acked_.push_back(AckedPacket(acked, 0, QuicTime::Zero()));
1300 if (acked == FirstSendingPacketNumber()) {
1301 break;
1302 }
1303 }
1304 if (acked_packets_iter_ == last_ack_frame_.packets.rend() ||
1305 start > acked_packets_iter_->min()) {
1306 // Finish adding all newly acked packets.
1307 return;
1308 }
1309 end = std::min(end, acked_packets_iter_->min());
1310 ++acked_packets_iter_;
1311 } while (start < end);
1312}
1313
1314void QuicSentPacketManager::OnAckTimestamp(QuicPacketNumber packet_number,
1315 QuicTime timestamp) {
1316 last_ack_frame_.received_packet_times.push_back({packet_number, timestamp});
1317 for (AckedPacket& packet : packets_acked_) {
1318 if (packet.packet_number == packet_number) {
1319 packet.receive_timestamp = timestamp;
1320 return;
1321 }
1322 }
1323}
1324
fayang3eb82212019-04-16 12:05:46 -07001325AckResult QuicSentPacketManager::OnAckFrameEnd(
1326 QuicTime ack_receive_time,
fayangf8e918b2019-07-16 13:03:16 -07001327 QuicPacketNumber ack_packet_number,
fayang3eb82212019-04-16 12:05:46 -07001328 EncryptionLevel ack_decrypted_level) {
QUICHE teama6ef0a62019-03-07 20:34:33 -05001329 QuicByteCount prior_bytes_in_flight = unacked_packets_.bytes_in_flight();
1330 // Reverse packets_acked_ so that it is in ascending order.
wubc0a1df42019-11-15 08:49:59 -08001331 std::reverse(packets_acked_.begin(), packets_acked_.end());
QUICHE teama6ef0a62019-03-07 20:34:33 -05001332 for (AckedPacket& acked_packet : packets_acked_) {
1333 QuicTransmissionInfo* info =
1334 unacked_packets_.GetMutableTransmissionInfo(acked_packet.packet_number);
1335 if (!QuicUtils::IsAckable(info->state)) {
1336 if (info->state == ACKED) {
1337 QUIC_BUG << "Trying to ack an already acked packet: "
1338 << acked_packet.packet_number
1339 << ", last_ack_frame_: " << last_ack_frame_
1340 << ", least_unacked: " << unacked_packets_.GetLeastUnacked()
1341 << ", packets_acked_: " << packets_acked_;
1342 } else {
fayang3eb82212019-04-16 12:05:46 -07001343 QUIC_PEER_BUG << "Received "
dschinazief79a5f2019-10-04 10:32:54 -07001344 << EncryptionLevelToString(ack_decrypted_level)
fayang3eb82212019-04-16 12:05:46 -07001345 << " ack for unackable packet: "
QUICHE teama6ef0a62019-03-07 20:34:33 -05001346 << acked_packet.packet_number << " with state: "
1347 << QuicUtils::SentPacketStateToString(info->state);
fayang3eb82212019-04-16 12:05:46 -07001348 if (supports_multiple_packet_number_spaces()) {
1349 if (info->state == NEVER_SENT) {
1350 return UNSENT_PACKETS_ACKED;
1351 }
1352 return UNACKABLE_PACKETS_ACKED;
1353 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001354 }
1355 continue;
1356 }
fayang3eb82212019-04-16 12:05:46 -07001357 QUIC_DVLOG(1) << ENDPOINT << "Got an "
dschinazief79a5f2019-10-04 10:32:54 -07001358 << EncryptionLevelToString(ack_decrypted_level)
fayang19d2d5b2019-09-11 14:22:03 -07001359 << " ack for packet " << acked_packet.packet_number
1360 << " , state: "
1361 << QuicUtils::SentPacketStateToString(info->state);
fayang3eb82212019-04-16 12:05:46 -07001362 const PacketNumberSpace packet_number_space =
fayangbf3d2862019-06-20 14:13:44 -07001363 unacked_packets_.GetPacketNumberSpace(info->encryption_level);
fayang3eb82212019-04-16 12:05:46 -07001364 if (supports_multiple_packet_number_spaces() &&
1365 QuicUtils::GetPacketNumberSpace(ack_decrypted_level) !=
1366 packet_number_space) {
1367 return PACKETS_ACKED_IN_WRONG_PACKET_NUMBER_SPACE;
1368 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001369 last_ack_frame_.packets.Add(acked_packet.packet_number);
fayang2f2915d2020-01-24 06:47:15 -08001370 if (info->encryption_level == ENCRYPTION_FORWARD_SECURE) {
1371 one_rtt_packet_acked_ = true;
1372 }
QUICHE teamc264e362019-03-19 14:21:06 -07001373 largest_packet_peer_knows_is_acked_.UpdateMax(info->largest_acked);
QUICHE teamc279cec2019-03-22 06:51:48 -07001374 if (supports_multiple_packet_number_spaces()) {
fayang3eb82212019-04-16 12:05:46 -07001375 largest_packets_peer_knows_is_acked_[packet_number_space].UpdateMax(
1376 info->largest_acked);
QUICHE teamc279cec2019-03-22 06:51:48 -07001377 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001378 // If data is associated with the most recent transmission of this
1379 // packet, then inform the caller.
1380 if (info->in_flight) {
1381 acked_packet.bytes_acked = info->bytes_sent;
1382 } else {
1383 // Unackable packets are skipped earlier.
1384 largest_newly_acked_ = acked_packet.packet_number;
1385 }
fayangbf3d2862019-06-20 14:13:44 -07001386 unacked_packets_.MaybeUpdateLargestAckedOfPacketNumberSpace(
1387 packet_number_space, acked_packet.packet_number);
fayang19d2d5b2019-09-11 14:22:03 -07001388 MarkPacketHandled(acked_packet.packet_number, info, ack_receive_time,
QUICHE team9467db02019-05-30 09:38:45 -07001389 last_ack_frame_.ack_delay_time,
1390 acked_packet.receive_timestamp);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001391 }
1392 const bool acked_new_packet = !packets_acked_.empty();
vasilvv923b6d22020-03-23 11:17:58 -07001393 PostProcessNewlyAckedPackets(ack_packet_number, ack_decrypted_level,
1394 last_ack_frame_, ack_receive_time, rtt_updated_,
fayang3eb82212019-04-16 12:05:46 -07001395 prior_bytes_in_flight);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001396
fayang3eb82212019-04-16 12:05:46 -07001397 return acked_new_packet ? PACKETS_NEWLY_ACKED : NO_PACKETS_NEWLY_ACKED;
QUICHE teama6ef0a62019-03-07 20:34:33 -05001398}
1399
1400void QuicSentPacketManager::SetDebugDelegate(DebugDelegate* debug_delegate) {
1401 debug_delegate_ = debug_delegate;
1402}
1403
1404void QuicSentPacketManager::OnApplicationLimited() {
1405 if (using_pacing_) {
1406 pacing_sender_.OnApplicationLimited();
1407 }
1408 send_algorithm_->OnApplicationLimited(unacked_packets_.bytes_in_flight());
1409 if (debug_delegate_ != nullptr) {
1410 debug_delegate_->OnApplicationLimited();
1411 }
1412}
1413
1414QuicTime QuicSentPacketManager::GetNextReleaseTime() const {
1415 return using_pacing_ ? pacing_sender_.ideal_next_packet_send_time()
1416 : QuicTime::Zero();
1417}
1418
1419void QuicSentPacketManager::SetInitialRtt(QuicTime::Delta rtt) {
1420 const QuicTime::Delta min_rtt =
1421 QuicTime::Delta::FromMicroseconds(kMinInitialRoundTripTimeUs);
1422 const QuicTime::Delta max_rtt =
1423 QuicTime::Delta::FromMicroseconds(kMaxInitialRoundTripTimeUs);
1424 rtt_stats_.set_initial_rtt(std::max(min_rtt, std::min(max_rtt, rtt)));
1425}
1426
QUICHE teamc279cec2019-03-22 06:51:48 -07001427void QuicSentPacketManager::EnableMultiplePacketNumberSpacesSupport() {
fayang18ff23b2020-01-28 09:19:00 -08001428 EnableIetfPtoAndLossDetection();
QUICHE teamc279cec2019-03-22 06:51:48 -07001429 unacked_packets_.EnableMultiplePacketNumberSpacesSupport();
1430}
1431
1432QuicPacketNumber QuicSentPacketManager::GetLargestAckedPacket(
1433 EncryptionLevel decrypted_packet_level) const {
1434 DCHECK(supports_multiple_packet_number_spaces());
1435 return unacked_packets_.GetLargestAckedOfPacketNumberSpace(
1436 QuicUtils::GetPacketNumberSpace(decrypted_packet_level));
1437}
1438
QUICHE teamc279cec2019-03-22 06:51:48 -07001439QuicPacketNumber QuicSentPacketManager::GetLargestPacketPeerKnowsIsAcked(
1440 EncryptionLevel decrypted_packet_level) const {
1441 DCHECK(supports_multiple_packet_number_spaces());
1442 return largest_packets_peer_knows_is_acked_[QuicUtils::GetPacketNumberSpace(
1443 decrypted_packet_level)];
1444}
1445
fayang1640cc22020-03-21 14:11:02 -07001446QuicTime::Delta
1447QuicSentPacketManager::GetNConsecutiveRetransmissionTimeoutDelay(
1448 int num_timeouts) const {
1449 QuicTime::Delta total_delay = QuicTime::Delta::Zero();
1450 const QuicTime::Delta srtt = rtt_stats_.SmoothedOrInitialRtt();
1451 int num_tlps =
1452 std::min(num_timeouts, static_cast<int>(max_tail_loss_probes_));
1453 num_timeouts -= num_tlps;
1454 if (num_tlps > 0) {
1455 if (enable_half_rtt_tail_loss_probe_ &&
1456 unacked_packets().HasUnackedStreamData()) {
1457 total_delay = total_delay + std::max(min_tlp_timeout_, srtt * 0.5);
1458 --num_tlps;
1459 }
1460 if (num_tlps > 0) {
1461 const QuicTime::Delta tlp_delay =
1462 std::max(2 * srtt, unacked_packets_.HasMultipleInFlightPackets()
1463 ? min_tlp_timeout_
1464 : (1.5 * srtt + (min_rto_timeout_ * 0.5)));
1465 total_delay = total_delay + num_tlps * tlp_delay;
1466 }
1467 }
1468 if (num_timeouts == 0) {
1469 return total_delay;
1470 }
1471
1472 const QuicTime::Delta retransmission_delay =
1473 rtt_stats_.smoothed_rtt().IsZero()
1474 ? QuicTime::Delta::FromMilliseconds(kDefaultRetransmissionTimeMs)
1475 : std::max(srtt + 4 * rtt_stats_.mean_deviation(), min_rto_timeout_);
1476 total_delay = total_delay + ((1 << num_timeouts) - 1) * retransmission_delay;
1477 return total_delay;
1478}
1479
QUICHE teama6ef0a62019-03-07 20:34:33 -05001480#undef ENDPOINT // undef for jumbo builds
1481} // namespace quic