blob: 2b52dd234575a8e03d473c212fde3fb1a1d3535b [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
35// Ensure the handshake timer isnt't faster than 10ms.
36// This limits the tenth retransmitted packet to 10s after the initial CHLO.
37static const int64_t kMinHandshakeTimeoutMs = 10;
38
39// Sends up to two tail loss probes before firing an RTO,
40// per draft RFC draft-dukkipati-tcpm-tcp-loss-probe.
41static const size_t kDefaultMaxTailLossProbes = 2;
42
QUICHE teama6ef0a62019-03-07 20:34:33 -050043// Returns true of retransmissions of the specified type should retransmit
44// the frames directly (as opposed to resulting in a loss notification).
45inline bool ShouldForceRetransmission(TransmissionType transmission_type) {
46 return transmission_type == HANDSHAKE_RETRANSMISSION ||
47 transmission_type == TLP_RETRANSMISSION ||
48 transmission_type == PROBING_RETRANSMISSION ||
fayang73d0ac42019-10-31 12:45:31 -070049 transmission_type == RTO_RETRANSMISSION ||
50 transmission_type == PTO_RETRANSMISSION;
QUICHE teama6ef0a62019-03-07 20:34:33 -050051}
52
fayang4ba55982019-05-13 05:53:22 -070053// If pacing rate is accurate, > 2 burst token is not likely to help first ACK
54// to arrive earlier, and overly large burst token could cause incast packet
55// losses.
56static const uint32_t kConservativeUnpacedBurst = 2;
57
QUICHE teama6ef0a62019-03-07 20:34:33 -050058} // namespace
59
60#define ENDPOINT \
61 (unacked_packets_.perspective() == Perspective::IS_SERVER ? "Server: " \
62 : "Client: ")
63
64QuicSentPacketManager::QuicSentPacketManager(
65 Perspective perspective,
66 const QuicClock* clock,
QUICHE team73957f12019-04-18 16:21:52 -070067 QuicRandom* random,
QUICHE teama6ef0a62019-03-07 20:34:33 -050068 QuicConnectionStats* stats,
69 CongestionControlType congestion_control_type,
70 LossDetectionType loss_type)
71 : unacked_packets_(perspective),
72 clock_(clock),
QUICHE team73957f12019-04-18 16:21:52 -070073 random_(random),
QUICHE teama6ef0a62019-03-07 20:34:33 -050074 stats_(stats),
75 debug_delegate_(nullptr),
76 network_change_visitor_(nullptr),
77 initial_congestion_window_(kInitialCongestionWindow),
wub37ec5962019-04-03 09:02:51 -070078 loss_algorithm_(GetInitialLossAlgorithm()),
QUICHE teama6ef0a62019-03-07 20:34:33 -050079 uber_loss_algorithm_(loss_type),
80 consecutive_rto_count_(0),
81 consecutive_tlp_count_(0),
82 consecutive_crypto_retransmission_count_(0),
83 pending_timer_transmission_count_(0),
84 max_tail_loss_probes_(kDefaultMaxTailLossProbes),
85 max_rto_packets_(kMaxRetransmissionsOnTimeout),
86 enable_half_rtt_tail_loss_probe_(false),
87 using_pacing_(false),
88 use_new_rto_(false),
89 conservative_handshake_retransmits_(false),
90 min_tlp_timeout_(
91 QuicTime::Delta::FromMilliseconds(kMinTailLossProbeTimeoutMs)),
92 min_rto_timeout_(
93 QuicTime::Delta::FromMilliseconds(kMinRetransmissionTimeMs)),
QUICHE teama6ef0a62019-03-07 20:34:33 -050094 largest_mtu_acked_(0),
95 handshake_confirmed_(false),
fkastenholz59c653b2019-07-15 09:55:53 -070096 peer_max_ack_delay_(
QUICHE teama6ef0a62019-03-07 20:34:33 -050097 QuicTime::Delta::FromMilliseconds(kDefaultDelayedAckTimeMs)),
98 rtt_updated_(false),
QUICHE team9929cc42019-03-13 08:17:43 -070099 acked_packets_iter_(last_ack_frame_.packets.rbegin()),
fayang62f867a2019-08-22 12:05:01 -0700100 pto_enabled_(false),
fayangce0a3162019-08-15 09:05:36 -0700101 max_probe_packets_per_pto_(2),
102 consecutive_pto_count_(0),
fayang19d2d5b2019-09-11 14:22:03 -0700103 handshake_mode_disabled_(false),
fayang70f9f932019-10-18 06:55:09 -0700104 forward_secure_packet_acked_(false),
vasilvv975c2352019-10-03 16:53:57 -0400105 neuter_handshake_packets_once_(
106 GetQuicReloadableFlag(quic_neuter_handshake_packets_once)) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500107 SetSendAlgorithm(congestion_control_type);
108}
109
wub37ec5962019-04-03 09:02:51 -0700110LossDetectionInterface* QuicSentPacketManager::GetInitialLossAlgorithm() {
fayangbf3d2862019-06-20 14:13:44 -0700111 return &uber_loss_algorithm_;
wub37ec5962019-04-03 09:02:51 -0700112}
113
QUICHE teama6ef0a62019-03-07 20:34:33 -0500114QuicSentPacketManager::~QuicSentPacketManager() {}
115
116void QuicSentPacketManager::SetFromConfig(const QuicConfig& config) {
117 const Perspective perspective = unacked_packets_.perspective();
118 if (config.HasReceivedInitialRoundTripTimeUs() &&
119 config.ReceivedInitialRoundTripTimeUs() > 0) {
120 if (!config.HasClientSentConnectionOption(kNRTT, perspective)) {
121 SetInitialRtt(QuicTime::Delta::FromMicroseconds(
122 config.ReceivedInitialRoundTripTimeUs()));
123 }
124 } else if (config.HasInitialRoundTripTimeUsToSend() &&
125 config.GetInitialRoundTripTimeUsToSend() > 0) {
126 SetInitialRtt(QuicTime::Delta::FromMicroseconds(
127 config.GetInitialRoundTripTimeUsToSend()));
128 }
ianswett43eefae2019-08-02 12:27:15 -0700129 if (config.HasReceivedMaxAckDelayMs()) {
130 peer_max_ack_delay_ =
131 QuicTime::Delta::FromMilliseconds(config.ReceivedMaxAckDelayMs());
132 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500133 if (config.HasClientSentConnectionOption(kMAD0, perspective)) {
134 rtt_stats_.set_ignore_max_ack_delay(true);
135 }
136 if (config.HasClientSentConnectionOption(kMAD1, perspective)) {
ianswett43eefae2019-08-02 12:27:15 -0700137 rtt_stats_.set_initial_max_ack_delay(peer_max_ack_delay_);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500138 }
ianswettad65ab92019-10-28 07:19:07 -0700139 if (config.HasClientSentConnectionOption(kMAD2, perspective)) {
140 // Set the minimum to the alarm granularity.
141 min_tlp_timeout_ = QuicTime::Delta::FromMilliseconds(1);
142 }
143 if (config.HasClientSentConnectionOption(kMAD3, perspective)) {
144 // Set the minimum to the alarm granularity.
145 min_rto_timeout_ = QuicTime::Delta::FromMilliseconds(1);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500146 }
147
fayangcff885a2019-10-22 07:39:04 -0700148 if (GetQuicReloadableFlag(quic_enable_pto)) {
fayangce0a3162019-08-15 09:05:36 -0700149 if (config.HasClientSentConnectionOption(k2PTO, perspective)) {
fayang62f867a2019-08-22 12:05:01 -0700150 pto_enabled_ = true;
fayangce0a3162019-08-15 09:05:36 -0700151 QUIC_RELOADABLE_FLAG_COUNT_N(quic_enable_pto, 2, 4);
152 }
153 if (config.HasClientSentConnectionOption(k1PTO, perspective)) {
fayang62f867a2019-08-22 12:05:01 -0700154 pto_enabled_ = true;
fayangce0a3162019-08-15 09:05:36 -0700155 max_probe_packets_per_pto_ = 1;
156 QUIC_RELOADABLE_FLAG_COUNT_N(quic_enable_pto, 1, 4);
157 }
158 }
159
QUICHE teama6ef0a62019-03-07 20:34:33 -0500160 // Configure congestion control.
161 if (config.HasClientRequestedIndependentOption(kTBBR, perspective)) {
162 SetSendAlgorithm(kBBR);
163 }
wuba9a43cb2019-07-17 15:22:42 -0700164 if (GetQuicReloadableFlag(quic_allow_client_enabled_bbr_v2) &&
165 config.HasClientRequestedIndependentOption(kB2ON, perspective)) {
166 QUIC_RELOADABLE_FLAG_COUNT(quic_allow_client_enabled_bbr_v2);
167 SetSendAlgorithm(kBBRv2);
168 }
169
QUICHE teama6ef0a62019-03-07 20:34:33 -0500170 if (config.HasClientRequestedIndependentOption(kRENO, perspective)) {
171 SetSendAlgorithm(kRenoBytes);
172 } else if (config.HasClientRequestedIndependentOption(kBYTE, perspective) ||
173 (GetQuicReloadableFlag(quic_default_to_bbr) &&
174 config.HasClientRequestedIndependentOption(kQBIC, perspective))) {
175 SetSendAlgorithm(kCubicBytes);
176 } else if (GetQuicReloadableFlag(quic_enable_pcc3) &&
177 config.HasClientRequestedIndependentOption(kTPCC, perspective)) {
178 SetSendAlgorithm(kPCC);
179 }
wuba9a43cb2019-07-17 15:22:42 -0700180
QUICHE teama6ef0a62019-03-07 20:34:33 -0500181 // Initial window.
182 if (GetQuicReloadableFlag(quic_unified_iw_options)) {
183 if (config.HasClientRequestedIndependentOption(kIW03, perspective)) {
184 initial_congestion_window_ = 3;
185 send_algorithm_->SetInitialCongestionWindowInPackets(3);
186 }
187 if (config.HasClientRequestedIndependentOption(kIW10, perspective)) {
188 initial_congestion_window_ = 10;
189 send_algorithm_->SetInitialCongestionWindowInPackets(10);
190 }
191 if (config.HasClientRequestedIndependentOption(kIW20, perspective)) {
192 initial_congestion_window_ = 20;
193 send_algorithm_->SetInitialCongestionWindowInPackets(20);
194 }
195 if (config.HasClientRequestedIndependentOption(kIW50, perspective)) {
196 initial_congestion_window_ = 50;
197 send_algorithm_->SetInitialCongestionWindowInPackets(50);
198 }
199 }
200
danzh88e3e052019-06-13 11:47:18 -0700201 using_pacing_ = !GetQuicFlag(FLAGS_quic_disable_pacing_for_perf_tests);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500202
QUICHE teama6ef0a62019-03-07 20:34:33 -0500203 if (config.HasClientSentConnectionOption(kNTLP, perspective)) {
204 max_tail_loss_probes_ = 0;
205 }
206 if (config.HasClientSentConnectionOption(k1TLP, perspective)) {
207 max_tail_loss_probes_ = 1;
208 }
209 if (config.HasClientSentConnectionOption(k1RTO, perspective)) {
210 max_rto_packets_ = 1;
211 }
212 if (config.HasClientSentConnectionOption(kTLPR, perspective)) {
213 enable_half_rtt_tail_loss_probe_ = true;
214 }
215 if (config.HasClientSentConnectionOption(kNRTO, perspective)) {
216 use_new_rto_ = true;
217 }
218 // Configure loss detection.
219 if (config.HasClientRequestedIndependentOption(kTIME, perspective)) {
fayangbf3d2862019-06-20 14:13:44 -0700220 uber_loss_algorithm_.SetLossDetectionType(kTime);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500221 }
222 if (config.HasClientRequestedIndependentOption(kATIM, perspective)) {
fayangbf3d2862019-06-20 14:13:44 -0700223 uber_loss_algorithm_.SetLossDetectionType(kAdaptiveTime);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500224 }
225 if (config.HasClientRequestedIndependentOption(kLFAK, perspective)) {
fayangbf3d2862019-06-20 14:13:44 -0700226 uber_loss_algorithm_.SetLossDetectionType(kLazyFack);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500227 }
fayangb0c7b4b2019-09-12 06:45:24 -0700228 if (GetQuicReloadableFlag(quic_enable_ietf_loss_detection)) {
229 if (config.HasClientRequestedIndependentOption(kILD0, perspective)) {
230 QUIC_RELOADABLE_FLAG_COUNT_N(quic_enable_ietf_loss_detection, 1, 4);
231 uber_loss_algorithm_.SetLossDetectionType(kIetfLossDetection);
232 }
233 if (config.HasClientRequestedIndependentOption(kILD1, perspective)) {
234 QUIC_RELOADABLE_FLAG_COUNT_N(quic_enable_ietf_loss_detection, 2, 4);
235 uber_loss_algorithm_.SetLossDetectionType(kIetfLossDetection);
236 uber_loss_algorithm_.SetReorderingShift(kDefaultLossDelayShift);
237 }
fayangcc4ea6a2019-10-25 08:44:03 -0700238 if (config.HasClientRequestedIndependentOption(kILD2, perspective)) {
239 QUIC_RELOADABLE_FLAG_COUNT_N(quic_enable_ietf_loss_detection, 3, 4);
240 uber_loss_algorithm_.SetLossDetectionType(kIetfLossDetection);
241 uber_loss_algorithm_.EnableAdaptiveReorderingThreshold();
242 }
243 if (config.HasClientRequestedIndependentOption(kILD3, perspective)) {
244 QUIC_RELOADABLE_FLAG_COUNT_N(quic_enable_ietf_loss_detection, 4, 4);
245 uber_loss_algorithm_.SetLossDetectionType(kIetfLossDetection);
246 uber_loss_algorithm_.SetReorderingShift(kDefaultLossDelayShift);
247 uber_loss_algorithm_.EnableAdaptiveReorderingThreshold();
fayangb0c7b4b2019-09-12 06:45:24 -0700248 }
249 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500250 if (config.HasClientSentConnectionOption(kCONH, perspective)) {
251 conservative_handshake_retransmits_ = true;
252 }
253 send_algorithm_->SetFromConfig(config, perspective);
254
255 if (network_change_visitor_ != nullptr) {
256 network_change_visitor_->OnCongestionChange();
257 }
258}
259
260void QuicSentPacketManager::ResumeConnectionState(
261 const CachedNetworkParameters& cached_network_params,
262 bool max_bandwidth_resumption) {
263 QuicBandwidth bandwidth = QuicBandwidth::FromBytesPerSecond(
264 max_bandwidth_resumption
265 ? cached_network_params.max_bandwidth_estimate_bytes_per_second()
266 : cached_network_params.bandwidth_estimate_bytes_per_second());
267 QuicTime::Delta rtt =
268 QuicTime::Delta::FromMilliseconds(cached_network_params.min_rtt_ms());
fayangf1b99dc2019-05-14 06:29:18 -0700269 AdjustNetworkParameters(bandwidth, rtt, /*allow_cwnd_to_decrease=*/false);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500270}
271
fayangf1b99dc2019-05-14 06:29:18 -0700272void QuicSentPacketManager::AdjustNetworkParameters(
273 QuicBandwidth bandwidth,
274 QuicTime::Delta rtt,
275 bool allow_cwnd_to_decrease) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500276 if (!rtt.IsZero()) {
277 SetInitialRtt(rtt);
278 }
fayangbe83ecd2019-04-26 13:58:09 -0700279 const QuicByteCount old_cwnd = send_algorithm_->GetCongestionWindow();
fayang6c2d64f2019-05-20 17:12:05 -0700280 if (GetQuicReloadableFlag(quic_conservative_bursts) && using_pacing_ &&
281 !bandwidth.IsZero()) {
fayang4ba55982019-05-13 05:53:22 -0700282 QUIC_RELOADABLE_FLAG_COUNT(quic_conservative_bursts);
283 pacing_sender_.SetBurstTokens(kConservativeUnpacedBurst);
284 }
fayangf1b99dc2019-05-14 06:29:18 -0700285 send_algorithm_->AdjustNetworkParameters(bandwidth, rtt,
286 allow_cwnd_to_decrease);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500287 if (debug_delegate_ != nullptr) {
fayangbe83ecd2019-04-26 13:58:09 -0700288 debug_delegate_->OnAdjustNetworkParameters(
fayangef9c8f92019-04-29 13:35:13 -0700289 bandwidth, rtt.IsZero() ? rtt_stats_.SmoothedOrInitialRtt() : rtt,
290 old_cwnd, send_algorithm_->GetCongestionWindow());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500291 }
292}
293
294void QuicSentPacketManager::SetHandshakeConfirmed() {
vasilvv975c2352019-10-03 16:53:57 -0400295 if (!neuter_handshake_packets_once_ || !handshake_confirmed_) {
296 if (neuter_handshake_packets_once_) {
297 QUIC_RELOADABLE_FLAG_COUNT(quic_neuter_handshake_packets_once);
298 }
299 handshake_confirmed_ = true;
300 NeuterHandshakePackets();
301 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500302}
303
fayang3eb82212019-04-16 12:05:46 -0700304void QuicSentPacketManager::PostProcessNewlyAckedPackets(
fayangf8e918b2019-07-16 13:03:16 -0700305 QuicPacketNumber ack_packet_number,
QUICHE teama6ef0a62019-03-07 20:34:33 -0500306 const QuicAckFrame& ack_frame,
307 QuicTime ack_receive_time,
308 bool rtt_updated,
309 QuicByteCount prior_bytes_in_flight) {
fayangcff885a2019-10-22 07:39:04 -0700310 unacked_packets_.NotifyAggregatedStreamFrameAcked(
311 last_ack_frame_.ack_delay_time);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500312 InvokeLossDetection(ack_receive_time);
313 // Ignore losses in RTO mode.
314 if (consecutive_rto_count_ > 0 && !use_new_rto_) {
315 packets_lost_.clear();
316 }
317 MaybeInvokeCongestionEvent(rtt_updated, prior_bytes_in_flight,
318 ack_receive_time);
319 unacked_packets_.RemoveObsoletePackets();
320
321 sustained_bandwidth_recorder_.RecordEstimate(
322 send_algorithm_->InRecovery(), send_algorithm_->InSlowStart(),
323 send_algorithm_->BandwidthEstimate(), ack_receive_time, clock_->WallNow(),
324 rtt_stats_.smoothed_rtt());
325
326 // Anytime we are making forward progress and have a new RTT estimate, reset
327 // the backoff counters.
328 if (rtt_updated) {
329 if (consecutive_rto_count_ > 0) {
330 // If the ack acknowledges data sent prior to the RTO,
331 // the RTO was spurious.
332 if (LargestAcked(ack_frame) < first_rto_transmission_) {
333 // Replace SRTT with latest_rtt and increase the variance to prevent
334 // a spurious RTO from happening again.
335 rtt_stats_.ExpireSmoothedMetrics();
336 } else {
337 if (!use_new_rto_) {
338 send_algorithm_->OnRetransmissionTimeout(true);
339 }
340 }
341 }
342 // Reset all retransmit counters any time a new packet is acked.
343 consecutive_rto_count_ = 0;
344 consecutive_tlp_count_ = 0;
fayangce0a3162019-08-15 09:05:36 -0700345 consecutive_pto_count_ = 0;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500346 consecutive_crypto_retransmission_count_ = 0;
347 }
348
349 if (debug_delegate_ != nullptr) {
fayangf8e918b2019-07-16 13:03:16 -0700350 debug_delegate_->OnIncomingAck(ack_packet_number, ack_frame,
351 ack_receive_time, LargestAcked(ack_frame),
352 rtt_updated, GetLeastUnacked());
QUICHE teama6ef0a62019-03-07 20:34:33 -0500353 }
354 // Remove packets below least unacked from all_packets_acked_ and
355 // last_ack_frame_.
356 last_ack_frame_.packets.RemoveUpTo(unacked_packets_.GetLeastUnacked());
357 last_ack_frame_.received_packet_times.clear();
358}
359
360void QuicSentPacketManager::MaybeInvokeCongestionEvent(
361 bool rtt_updated,
362 QuicByteCount prior_in_flight,
363 QuicTime event_time) {
364 if (!rtt_updated && packets_acked_.empty() && packets_lost_.empty()) {
365 return;
366 }
367 if (using_pacing_) {
368 pacing_sender_.OnCongestionEvent(rtt_updated, prior_in_flight, event_time,
369 packets_acked_, packets_lost_);
370 } else {
371 send_algorithm_->OnCongestionEvent(rtt_updated, prior_in_flight, event_time,
372 packets_acked_, packets_lost_);
373 }
374 packets_acked_.clear();
375 packets_lost_.clear();
376 if (network_change_visitor_ != nullptr) {
377 network_change_visitor_->OnCongestionChange();
378 }
379}
380
381void QuicSentPacketManager::RetransmitUnackedPackets(
382 TransmissionType retransmission_type) {
383 DCHECK(retransmission_type == ALL_UNACKED_RETRANSMISSION ||
384 retransmission_type == ALL_INITIAL_RETRANSMISSION);
385 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
ianswett898306b2019-04-23 11:05:57 -0700386 for (QuicUnackedPacketMap::iterator it = unacked_packets_.begin();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500387 it != unacked_packets_.end(); ++it, ++packet_number) {
388 if ((retransmission_type == ALL_UNACKED_RETRANSMISSION ||
ianswett898306b2019-04-23 11:05:57 -0700389 it->encryption_level == ENCRYPTION_ZERO_RTT)) {
ianswett04004652019-09-03 18:46:57 -0700390 if (it->in_flight) {
ianswett898306b2019-04-23 11:05:57 -0700391 // Remove 0-RTT packets and packets of the wrong version from flight,
392 // because neither can be processed by the peer.
393 unacked_packets_.RemoveFromInFlight(&*it);
394 }
395 if (unacked_packets_.HasRetransmittableFrames(*it)) {
396 MarkForRetransmission(packet_number, retransmission_type);
397 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500398 }
399 }
ianswett898306b2019-04-23 11:05:57 -0700400 if (retransmission_type == ALL_UNACKED_RETRANSMISSION &&
401 unacked_packets_.bytes_in_flight() > 0) {
402 QUIC_BUG << "RetransmitUnackedPackets should remove all packets from flight"
403 << ", bytes_in_flight:" << unacked_packets_.bytes_in_flight();
404 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500405}
406
407void QuicSentPacketManager::NeuterUnencryptedPackets() {
408 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500409 for (QuicUnackedPacketMap::const_iterator it = unacked_packets_.begin();
410 it != unacked_packets_.end(); ++it, ++packet_number) {
fayangcff885a2019-10-22 07:39:04 -0700411 if (!it->retransmittable_frames.empty() &&
412 it->encryption_level == ENCRYPTION_INITIAL) {
413 // Once the connection swithes to forward secure, no unencrypted packets
414 // will be sent. The data has been abandoned in the cryto stream. Remove
415 // it from in flight.
ianswett04004652019-09-03 18:46:57 -0700416 unacked_packets_.RemoveFromInFlight(packet_number);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500417 }
418 }
419}
420
421void QuicSentPacketManager::NeuterHandshakePackets() {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500422 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
423 for (QuicUnackedPacketMap::const_iterator it = unacked_packets_.begin();
424 it != unacked_packets_.end(); ++it, ++packet_number) {
fayangcff885a2019-10-22 07:39:04 -0700425 if (!it->retransmittable_frames.empty() &&
426 unacked_packets_.GetPacketNumberSpace(it->encryption_level) ==
427 HANDSHAKE_DATA) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500428 unacked_packets_.RemoveFromInFlight(packet_number);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500429 }
430 }
431}
432
433void QuicSentPacketManager::MarkForRetransmission(
434 QuicPacketNumber packet_number,
435 TransmissionType transmission_type) {
436 QuicTransmissionInfo* transmission_info =
437 unacked_packets_.GetMutableTransmissionInfo(packet_number);
fayangcff885a2019-10-22 07:39:04 -0700438 // A previous RTO retransmission may cause connection close; packets without
439 // retransmittable frames can be marked for loss retransmissions.
440 QUIC_BUG_IF(transmission_type != LOSS_RETRANSMISSION &&
441 transmission_type != RTO_RETRANSMISSION &&
QUICHE teama6ef0a62019-03-07 20:34:33 -0500442 !unacked_packets_.HasRetransmittableFrames(*transmission_info))
dschinazief79a5f2019-10-04 10:32:54 -0700443 << "transmission_type: " << TransmissionTypeToString(transmission_type);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500444 // Handshake packets should never be sent as probing retransmissions.
fayang73d0ac42019-10-31 12:45:31 -0700445 DCHECK(!transmission_info->has_crypto_handshake ||
QUICHE teama6ef0a62019-03-07 20:34:33 -0500446 transmission_type != PROBING_RETRANSMISSION);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500447
448 HandleRetransmission(transmission_type, transmission_info);
449
450 // Update packet state according to transmission type.
451 transmission_info->state =
452 QuicUtils::RetransmissionTypeToPacketState(transmission_type);
453}
454
455void QuicSentPacketManager::HandleRetransmission(
456 TransmissionType transmission_type,
457 QuicTransmissionInfo* transmission_info) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500458 if (ShouldForceRetransmission(transmission_type)) {
459 // TODO(fayang): Consider to make RTO and PROBING retransmission
460 // strategies be configurable by applications. Today, TLP, RTO and PROBING
461 // retransmissions are handled similarly, i.e., always retranmist the
462 // oldest outstanding data. This is not ideal in general because different
463 // applications may want different strategies. For example, some
464 // applications may want to use higher priority stream data for bandwidth
465 // probing, and some applications want to consider RTO is an indication of
466 // loss, etc.
467 unacked_packets_.RetransmitFrames(*transmission_info, transmission_type);
468 return;
469 }
470
471 unacked_packets_.NotifyFramesLost(*transmission_info, transmission_type);
472 if (transmission_info->retransmittable_frames.empty()) {
473 return;
474 }
475
476 if (transmission_type == LOSS_RETRANSMISSION) {
477 // Record the first packet sent after loss, which allows to wait 1
478 // more RTT before giving up on this lost packet.
479 transmission_info->retransmission =
480 unacked_packets_.largest_sent_packet() + 1;
481 } else {
482 // Clear the recorded first packet sent after loss when version or
483 // encryption changes.
484 transmission_info->retransmission.Clear();
485 }
486}
487
488void QuicSentPacketManager::RecordOneSpuriousRetransmission(
489 const QuicTransmissionInfo& info) {
490 stats_->bytes_spuriously_retransmitted += info.bytes_sent;
491 ++stats_->packets_spuriously_retransmitted;
492 if (debug_delegate_ != nullptr) {
493 debug_delegate_->OnSpuriousPacketRetransmission(info.transmission_type,
494 info.bytes_sent);
495 }
496}
497
QUICHE teama6ef0a62019-03-07 20:34:33 -0500498void QuicSentPacketManager::MarkPacketHandled(QuicPacketNumber packet_number,
499 QuicTransmissionInfo* info,
fayang19d2d5b2019-09-11 14:22:03 -0700500 QuicTime ack_receive_time,
QUICHE team9467db02019-05-30 09:38:45 -0700501 QuicTime::Delta ack_delay_time,
502 QuicTime receive_timestamp) {
fayangcff885a2019-10-22 07:39:04 -0700503 // Try to aggregate acked stream frames if acked packet is not a
504 // retransmission.
505 if (info->transmission_type == NOT_RETRANSMISSION) {
506 unacked_packets_.MaybeAggregateAckedStreamFrame(*info, ack_delay_time,
507 receive_timestamp);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500508 } else {
fayangcff885a2019-10-22 07:39:04 -0700509 unacked_packets_.NotifyAggregatedStreamFrameAcked(ack_delay_time);
510 const bool new_data_acked = unacked_packets_.NotifyFramesAcked(
511 *info, ack_delay_time, receive_timestamp);
512 if (!new_data_acked && info->transmission_type != NOT_RETRANSMISSION) {
513 // Record as a spurious retransmission if this packet is a
514 // retransmission and no new data gets acked.
515 QUIC_DVLOG(1) << "Detect spurious retransmitted packet " << packet_number
516 << " transmission type: "
517 << TransmissionTypeToString(info->transmission_type);
fayangcc4ea6a2019-10-25 08:44:03 -0700518 RecordOneSpuriousRetransmission(*info);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500519 }
520 }
fayangcc4ea6a2019-10-25 08:44:03 -0700521 if (info->state == LOST) {
fayangcff885a2019-10-22 07:39:04 -0700522 // Record as a spurious loss as a packet previously declared lost gets
523 // acked.
fayangcff885a2019-10-22 07:39:04 -0700524 const PacketNumberSpace packet_number_space =
525 unacked_packets_.GetPacketNumberSpace(info->encryption_level);
526 const QuicPacketNumber previous_largest_acked =
527 supports_multiple_packet_number_spaces()
528 ? unacked_packets_.GetLargestAckedOfPacketNumberSpace(
529 packet_number_space)
530 : unacked_packets_.largest_acked();
531 QUIC_DVLOG(1) << "Packet " << packet_number
532 << " was detected lost spuriously, "
533 "previous_largest_acked: "
534 << previous_largest_acked;
535 loss_algorithm_->SpuriousLossDetected(unacked_packets_, rtt_stats_,
536 ack_receive_time, packet_number,
537 previous_largest_acked);
538 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500539
540 if (network_change_visitor_ != nullptr &&
541 info->bytes_sent > largest_mtu_acked_) {
542 largest_mtu_acked_ = info->bytes_sent;
543 network_change_visitor_->OnPathMtuIncreased(largest_mtu_acked_);
544 }
545 unacked_packets_.RemoveFromInFlight(info);
546 unacked_packets_.RemoveRetransmittability(info);
547 info->state = ACKED;
548}
549
550bool QuicSentPacketManager::OnPacketSent(
551 SerializedPacket* serialized_packet,
QUICHE teama6ef0a62019-03-07 20:34:33 -0500552 QuicTime sent_time,
553 TransmissionType transmission_type,
554 HasRetransmittableData has_retransmittable_data) {
555 QuicPacketNumber packet_number = serialized_packet->packet_number;
556 DCHECK_LE(FirstSendingPacketNumber(), packet_number);
557 DCHECK(!unacked_packets_.IsUnacked(packet_number));
558 QUIC_BUG_IF(serialized_packet->encrypted_length == 0)
559 << "Cannot send empty packets.";
QUICHE teama6ef0a62019-03-07 20:34:33 -0500560 if (pending_timer_transmission_count_ > 0) {
561 --pending_timer_transmission_count_;
562 }
563
564 bool in_flight = has_retransmittable_data == HAS_RETRANSMITTABLE_DATA;
565 if (using_pacing_) {
566 pacing_sender_.OnPacketSent(
567 sent_time, unacked_packets_.bytes_in_flight(), packet_number,
568 serialized_packet->encrypted_length, has_retransmittable_data);
569 } else {
570 send_algorithm_->OnPacketSent(
571 sent_time, unacked_packets_.bytes_in_flight(), packet_number,
572 serialized_packet->encrypted_length, has_retransmittable_data);
573 }
574
fayangcff885a2019-10-22 07:39:04 -0700575 unacked_packets_.AddSentPacket(serialized_packet, transmission_type,
576 sent_time, in_flight);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500577 // Reset the retransmission timer anytime a pending packet is sent.
578 return in_flight;
579}
580
fayang67f82272019-08-14 16:08:45 -0700581QuicSentPacketManager::RetransmissionTimeoutMode
582QuicSentPacketManager::OnRetransmissionTimeout() {
fayang5f135052019-08-22 17:59:40 -0700583 DCHECK(unacked_packets_.HasInFlightPackets() ||
584 (handshake_mode_disabled_ && !handshake_confirmed_));
QUICHE teama6ef0a62019-03-07 20:34:33 -0500585 DCHECK_EQ(0u, pending_timer_transmission_count_);
586 // Handshake retransmission, timer based loss detection, TLP, and RTO are
587 // implemented with a single alarm. The handshake alarm is set when the
588 // handshake has not completed, the loss alarm is set when the loss detection
589 // algorithm says to, and the TLP and RTO alarms are set after that.
590 // The TLP alarm is always set to run for under an RTO.
591 switch (GetRetransmissionMode()) {
592 case HANDSHAKE_MODE:
fayang5f135052019-08-22 17:59:40 -0700593 DCHECK(!handshake_mode_disabled_);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500594 ++stats_->crypto_retransmit_count;
595 RetransmitCryptoPackets();
fayang67f82272019-08-14 16:08:45 -0700596 return HANDSHAKE_MODE;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500597 case LOSS_MODE: {
598 ++stats_->loss_timeout_count;
599 QuicByteCount prior_in_flight = unacked_packets_.bytes_in_flight();
600 const QuicTime now = clock_->Now();
601 InvokeLossDetection(now);
602 MaybeInvokeCongestionEvent(false, prior_in_flight, now);
fayang67f82272019-08-14 16:08:45 -0700603 return LOSS_MODE;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500604 }
605 case TLP_MODE:
606 ++stats_->tlp_count;
607 ++consecutive_tlp_count_;
608 pending_timer_transmission_count_ = 1;
609 // TLPs prefer sending new data instead of retransmitting data, so
610 // give the connection a chance to write before completing the TLP.
fayang67f82272019-08-14 16:08:45 -0700611 return TLP_MODE;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500612 case RTO_MODE:
613 ++stats_->rto_count;
614 RetransmitRtoPackets();
fayang67f82272019-08-14 16:08:45 -0700615 return RTO_MODE;
fayangce0a3162019-08-15 09:05:36 -0700616 case PTO_MODE:
617 QUIC_DVLOG(1) << ENDPOINT << "PTO mode";
618 ++stats_->pto_count;
619 ++consecutive_pto_count_;
620 pending_timer_transmission_count_ = max_probe_packets_per_pto_;
621 return PTO_MODE;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500622 }
623}
624
625void QuicSentPacketManager::RetransmitCryptoPackets() {
626 DCHECK_EQ(HANDSHAKE_MODE, GetRetransmissionMode());
627 ++consecutive_crypto_retransmission_count_;
628 bool packet_retransmitted = false;
629 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
630 std::vector<QuicPacketNumber> crypto_retransmissions;
631 for (QuicUnackedPacketMap::const_iterator it = unacked_packets_.begin();
632 it != unacked_packets_.end(); ++it, ++packet_number) {
633 // Only retransmit frames which are in flight, and therefore have been sent.
fayangcff885a2019-10-22 07:39:04 -0700634 if (!it->in_flight || it->state != OUTSTANDING ||
QUICHE teama6ef0a62019-03-07 20:34:33 -0500635 !it->has_crypto_handshake ||
636 !unacked_packets_.HasRetransmittableFrames(*it)) {
637 continue;
638 }
639 packet_retransmitted = true;
fayangcff885a2019-10-22 07:39:04 -0700640 crypto_retransmissions.push_back(packet_number);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500641 ++pending_timer_transmission_count_;
642 }
643 DCHECK(packet_retransmitted) << "No crypto packets found to retransmit.";
fayangcff885a2019-10-22 07:39:04 -0700644 for (QuicPacketNumber retransmission : crypto_retransmissions) {
645 MarkForRetransmission(retransmission, HANDSHAKE_RETRANSMISSION);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500646 }
647}
648
649bool QuicSentPacketManager::MaybeRetransmitTailLossProbe() {
fayang62f867a2019-08-22 12:05:01 -0700650 DCHECK(!pto_enabled_);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500651 if (pending_timer_transmission_count_ == 0) {
652 return false;
653 }
654 if (!MaybeRetransmitOldestPacket(TLP_RETRANSMISSION)) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500655 return false;
656 }
657 return true;
658}
659
660bool QuicSentPacketManager::MaybeRetransmitOldestPacket(TransmissionType type) {
661 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
662 for (QuicUnackedPacketMap::const_iterator it = unacked_packets_.begin();
663 it != unacked_packets_.end(); ++it, ++packet_number) {
664 // Only retransmit frames which are in flight, and therefore have been sent.
fayangcff885a2019-10-22 07:39:04 -0700665 if (!it->in_flight || it->state != OUTSTANDING ||
QUICHE teama6ef0a62019-03-07 20:34:33 -0500666 !unacked_packets_.HasRetransmittableFrames(*it)) {
667 continue;
668 }
669 MarkForRetransmission(packet_number, type);
670 return true;
671 }
672 QUIC_DVLOG(1)
673 << "No retransmittable packets, so RetransmitOldestPacket failed.";
674 return false;
675}
676
677void QuicSentPacketManager::RetransmitRtoPackets() {
fayang62f867a2019-08-22 12:05:01 -0700678 DCHECK(!pto_enabled_);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500679 QUIC_BUG_IF(pending_timer_transmission_count_ > 0)
680 << "Retransmissions already queued:" << pending_timer_transmission_count_;
681 // Mark two packets for retransmission.
682 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
683 std::vector<QuicPacketNumber> retransmissions;
684 for (QuicUnackedPacketMap::const_iterator it = unacked_packets_.begin();
685 it != unacked_packets_.end(); ++it, ++packet_number) {
fayangcff885a2019-10-22 07:39:04 -0700686 if (it->state == OUTSTANDING &&
QUICHE teama6ef0a62019-03-07 20:34:33 -0500687 unacked_packets_.HasRetransmittableFrames(*it) &&
688 pending_timer_transmission_count_ < max_rto_packets_) {
fayangcff885a2019-10-22 07:39:04 -0700689 retransmissions.push_back(packet_number);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500690 ++pending_timer_transmission_count_;
691 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500692 }
693 if (pending_timer_transmission_count_ > 0) {
694 if (consecutive_rto_count_ == 0) {
695 first_rto_transmission_ = unacked_packets_.largest_sent_packet() + 1;
696 }
697 ++consecutive_rto_count_;
698 }
fayangcff885a2019-10-22 07:39:04 -0700699 for (QuicPacketNumber retransmission : retransmissions) {
700 MarkForRetransmission(retransmission, RTO_RETRANSMISSION);
701 }
702 if (retransmissions.empty()) {
703 QUIC_BUG_IF(pending_timer_transmission_count_ != 0);
704 // No packets to be RTO retransmitted, raise up a credit to allow
705 // connection to send.
706 QUIC_CODE_COUNT(no_packets_to_be_rto_retransmitted);
707 pending_timer_transmission_count_ = 1;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500708 }
709}
710
fayangce0a3162019-08-15 09:05:36 -0700711void QuicSentPacketManager::MaybeSendProbePackets() {
712 if (pending_timer_transmission_count_ == 0) {
713 return;
714 }
715 QuicPacketNumber packet_number = unacked_packets_.GetLeastUnacked();
716 std::vector<QuicPacketNumber> probing_packets;
717 for (QuicUnackedPacketMap::const_iterator it = unacked_packets_.begin();
718 it != unacked_packets_.end(); ++it, ++packet_number) {
719 if (it->state == OUTSTANDING &&
720 unacked_packets_.HasRetransmittableFrames(*it)) {
721 probing_packets.push_back(packet_number);
722 if (probing_packets.size() == pending_timer_transmission_count_) {
723 break;
724 }
725 }
726 }
727
728 for (QuicPacketNumber retransmission : probing_packets) {
729 QUIC_DVLOG(1) << ENDPOINT << "Marking " << retransmission
730 << " for probing retransmission";
fayang73d0ac42019-10-31 12:45:31 -0700731 MarkForRetransmission(retransmission, PTO_RETRANSMISSION);
fayangce0a3162019-08-15 09:05:36 -0700732 }
733 // It is possible that there is not enough outstanding data for probing.
734}
735
736void QuicSentPacketManager::AdjustPendingTimerTransmissions() {
737 if (pending_timer_transmission_count_ < max_probe_packets_per_pto_) {
738 // There are packets sent already, clear credit.
739 pending_timer_transmission_count_ = 0;
740 return;
741 }
742 // No packet gets sent, leave 1 credit to allow data to be write eventually.
743 pending_timer_transmission_count_ = 1;
744}
745
fayang08a6c952019-09-18 14:29:45 -0700746void QuicSentPacketManager::EnableIetfPtoAndLossDetection() {
fayang5f135052019-08-22 17:59:40 -0700747 pto_enabled_ = true;
748 handshake_mode_disabled_ = true;
fayang08a6c952019-09-18 14:29:45 -0700749 uber_loss_algorithm_.SetLossDetectionType(kIetfLossDetection);
fayang5f135052019-08-22 17:59:40 -0700750}
751
QUICHE teama6ef0a62019-03-07 20:34:33 -0500752QuicSentPacketManager::RetransmissionTimeoutMode
753QuicSentPacketManager::GetRetransmissionMode() const {
fayang5f135052019-08-22 17:59:40 -0700754 DCHECK(unacked_packets_.HasInFlightPackets() ||
755 (handshake_mode_disabled_ && !handshake_confirmed_));
756 if (!handshake_mode_disabled_ && !handshake_confirmed_ &&
757 unacked_packets_.HasPendingCryptoPackets()) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500758 return HANDSHAKE_MODE;
759 }
760 if (loss_algorithm_->GetLossTimeout() != QuicTime::Zero()) {
761 return LOSS_MODE;
762 }
fayang62f867a2019-08-22 12:05:01 -0700763 if (pto_enabled_) {
fayangce0a3162019-08-15 09:05:36 -0700764 return PTO_MODE;
765 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500766 if (consecutive_tlp_count_ < max_tail_loss_probes_) {
ianswett7ddb34f2019-07-15 12:17:43 -0700767 if (unacked_packets_.HasUnackedRetransmittableFrames()) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500768 return TLP_MODE;
769 }
770 }
771 return RTO_MODE;
772}
773
774void QuicSentPacketManager::InvokeLossDetection(QuicTime time) {
775 if (!packets_acked_.empty()) {
776 DCHECK_LE(packets_acked_.front().packet_number,
777 packets_acked_.back().packet_number);
778 largest_newly_acked_ = packets_acked_.back().packet_number;
779 }
780 loss_algorithm_->DetectLosses(unacked_packets_, time, rtt_stats_,
781 largest_newly_acked_, packets_acked_,
782 &packets_lost_);
783 for (const LostPacket& packet : packets_lost_) {
784 ++stats_->packets_lost;
785 if (debug_delegate_ != nullptr) {
786 debug_delegate_->OnPacketLoss(packet.packet_number, LOSS_RETRANSMISSION,
787 time);
788 }
ianswett04004652019-09-03 18:46:57 -0700789 unacked_packets_.RemoveFromInFlight(packet.packet_number);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500790 MarkForRetransmission(packet.packet_number, LOSS_RETRANSMISSION);
791 }
792}
793
794bool QuicSentPacketManager::MaybeUpdateRTT(QuicPacketNumber largest_acked,
795 QuicTime::Delta ack_delay_time,
796 QuicTime ack_receive_time) {
797 // We rely on ack_delay_time to compute an RTT estimate, so we
798 // only update rtt when the largest observed gets acked.
799 if (!unacked_packets_.IsUnacked(largest_acked)) {
800 return false;
801 }
802 // We calculate the RTT based on the highest ACKed packet number, the lower
803 // packet numbers will include the ACK aggregation delay.
804 const QuicTransmissionInfo& transmission_info =
805 unacked_packets_.GetTransmissionInfo(largest_acked);
806 // Ensure the packet has a valid sent time.
807 if (transmission_info.sent_time == QuicTime::Zero()) {
808 QUIC_BUG << "Acked packet has zero sent time, largest_acked:"
809 << largest_acked;
810 return false;
811 }
812 if (transmission_info.sent_time > ack_receive_time) {
813 QUIC_CODE_COUNT(quic_receive_acked_before_sending);
814 }
815
816 QuicTime::Delta send_delta = ack_receive_time - transmission_info.sent_time;
817 rtt_stats_.UpdateRtt(send_delta, ack_delay_time, ack_receive_time);
818
819 return true;
820}
821
822QuicTime::Delta QuicSentPacketManager::TimeUntilSend(QuicTime now) const {
823 // The TLP logic is entirely contained within QuicSentPacketManager, so the
824 // send algorithm does not need to be consulted.
825 if (pending_timer_transmission_count_ > 0) {
826 return QuicTime::Delta::Zero();
827 }
828
829 if (using_pacing_) {
830 return pacing_sender_.TimeUntilSend(now,
831 unacked_packets_.bytes_in_flight());
832 }
833
834 return send_algorithm_->CanSend(unacked_packets_.bytes_in_flight())
835 ? QuicTime::Delta::Zero()
836 : QuicTime::Delta::Infinite();
837}
838
839const QuicTime QuicSentPacketManager::GetRetransmissionTime() const {
fayang5f135052019-08-22 17:59:40 -0700840 if (!unacked_packets_.HasInFlightPackets() &&
841 (!handshake_mode_disabled_ || handshake_confirmed_ ||
842 unacked_packets_.perspective() == Perspective::IS_SERVER)) {
843 // Do not set the timer if there is nothing in flight. However, to avoid
844 // handshake deadlock due to anti-amplification limit, client needs to set
845 // PTO timer when the handshake is not confirmed even there is nothing in
846 // flight.
847 return QuicTime::Zero();
848 }
849 if (pending_timer_transmission_count_ > 0) {
850 // Do not set the timer if there is any credit left.
QUICHE teama6ef0a62019-03-07 20:34:33 -0500851 return QuicTime::Zero();
852 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500853 switch (GetRetransmissionMode()) {
854 case HANDSHAKE_MODE:
855 return unacked_packets_.GetLastCryptoPacketSentTime() +
856 GetCryptoRetransmissionDelay();
857 case LOSS_MODE:
858 return loss_algorithm_->GetLossTimeout();
859 case TLP_MODE: {
fayang62f867a2019-08-22 12:05:01 -0700860 DCHECK(!pto_enabled_);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500861 // TODO(ianswett): When CWND is available, it would be preferable to
862 // set the timer based on the earliest retransmittable packet.
863 // Base the updated timer on the send time of the last packet.
ianswett479fea32019-09-04 02:51:01 -0700864 const QuicTime sent_time =
865 unacked_packets_.GetLastInFlightPacketSentTime();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500866 const QuicTime tlp_time = sent_time + GetTailLossProbeDelay();
867 // Ensure the TLP timer never gets set to a time in the past.
868 return std::max(clock_->ApproximateNow(), tlp_time);
869 }
870 case RTO_MODE: {
fayang62f867a2019-08-22 12:05:01 -0700871 DCHECK(!pto_enabled_);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500872 // The RTO is based on the first outstanding packet.
ianswett479fea32019-09-04 02:51:01 -0700873 const QuicTime sent_time =
874 unacked_packets_.GetLastInFlightPacketSentTime();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500875 QuicTime rto_time = sent_time + GetRetransmissionDelay();
876 // Wait for TLP packets to be acked before an RTO fires.
ianswett479fea32019-09-04 02:51:01 -0700877 QuicTime tlp_time = sent_time + GetTailLossProbeDelay();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500878 return std::max(tlp_time, rto_time);
879 }
fayangce0a3162019-08-15 09:05:36 -0700880 case PTO_MODE: {
881 // Ensure PTO never gets set to a time in the past.
ianswett479fea32019-09-04 02:51:01 -0700882 return std::max(clock_->ApproximateNow(),
883 unacked_packets_.GetLastInFlightPacketSentTime() +
884 GetProbeTimeoutDelay());
fayangce0a3162019-08-15 09:05:36 -0700885 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500886 }
887 DCHECK(false);
888 return QuicTime::Zero();
889}
890
891const QuicTime::Delta QuicSentPacketManager::GetPathDegradingDelay() const {
892 QuicTime::Delta delay = QuicTime::Delta::Zero();
893 for (size_t i = 0; i < max_tail_loss_probes_; ++i) {
894 delay = delay + GetTailLossProbeDelay(i);
895 }
896 for (size_t i = 0; i < kNumRetransmissionDelaysForPathDegradingDelay; ++i) {
897 delay = delay + GetRetransmissionDelay(i);
898 }
899 return delay;
900}
901
902const QuicTime::Delta QuicSentPacketManager::GetCryptoRetransmissionDelay()
903 const {
904 // This is equivalent to the TailLossProbeDelay, but slightly more aggressive
905 // because crypto handshake messages don't incur a delayed ack time.
906 QuicTime::Delta srtt = rtt_stats_.SmoothedOrInitialRtt();
907 int64_t delay_ms;
908 if (conservative_handshake_retransmits_) {
909 // Using the delayed ack time directly could cause conservative handshake
910 // retransmissions to actually be more aggressive than the default.
fkastenholz59c653b2019-07-15 09:55:53 -0700911 delay_ms = std::max(peer_max_ack_delay_.ToMilliseconds(),
QUICHE teama6ef0a62019-03-07 20:34:33 -0500912 static_cast<int64_t>(2 * srtt.ToMilliseconds()));
913 } else {
914 delay_ms = std::max(kMinHandshakeTimeoutMs,
915 static_cast<int64_t>(1.5 * srtt.ToMilliseconds()));
916 }
917 return QuicTime::Delta::FromMilliseconds(
918 delay_ms << consecutive_crypto_retransmission_count_);
919}
920
921const QuicTime::Delta QuicSentPacketManager::GetTailLossProbeDelay(
922 size_t consecutive_tlp_count) const {
923 QuicTime::Delta srtt = rtt_stats_.SmoothedOrInitialRtt();
924 if (enable_half_rtt_tail_loss_probe_ && consecutive_tlp_count == 0u) {
zhongyi1b2f7832019-06-14 13:31:34 -0700925 if (unacked_packets().HasUnackedStreamData()) {
926 // Enable TLPR if there are pending data packets.
927 return std::max(min_tlp_timeout_, srtt * 0.5);
928 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500929 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500930 if (!unacked_packets_.HasMultipleInFlightPackets()) {
931 // This expression really should be using the delayed ack time, but in TCP
932 // MinRTO was traditionally set to 2x the delayed ack timer and this
933 // expression assumed QUIC did the same.
934 return std::max(2 * srtt, 1.5 * srtt + (min_rto_timeout_ * 0.5));
935 }
936 return std::max(min_tlp_timeout_, 2 * srtt);
937}
938
939const QuicTime::Delta QuicSentPacketManager::GetRetransmissionDelay(
940 size_t consecutive_rto_count) const {
941 QuicTime::Delta retransmission_delay = QuicTime::Delta::Zero();
942 if (rtt_stats_.smoothed_rtt().IsZero()) {
943 // We are in the initial state, use default timeout values.
944 retransmission_delay =
945 QuicTime::Delta::FromMilliseconds(kDefaultRetransmissionTimeMs);
946 } else {
947 retransmission_delay =
948 rtt_stats_.smoothed_rtt() + 4 * rtt_stats_.mean_deviation();
949 if (retransmission_delay < min_rto_timeout_) {
950 retransmission_delay = min_rto_timeout_;
951 }
952 }
953
954 // Calculate exponential back off.
955 retransmission_delay =
956 retransmission_delay *
957 (1 << std::min<size_t>(consecutive_rto_count, kMaxRetransmissions));
958
959 if (retransmission_delay.ToMilliseconds() > kMaxRetransmissionTimeMs) {
960 return QuicTime::Delta::FromMilliseconds(kMaxRetransmissionTimeMs);
961 }
962 return retransmission_delay;
963}
964
fayangce0a3162019-08-15 09:05:36 -0700965const QuicTime::Delta QuicSentPacketManager::GetProbeTimeoutDelay() const {
fayang62f867a2019-08-22 12:05:01 -0700966 DCHECK(pto_enabled_);
fayangce0a3162019-08-15 09:05:36 -0700967 if (rtt_stats_.smoothed_rtt().IsZero()) {
968 if (rtt_stats_.initial_rtt().IsZero()) {
969 return QuicTime::Delta::FromSeconds(1);
970 }
971 return 2 * rtt_stats_.initial_rtt();
972 }
973 const QuicTime::Delta pto_delay =
974 rtt_stats_.smoothed_rtt() +
975 std::max(4 * rtt_stats_.mean_deviation(),
976 QuicTime::Delta::FromMilliseconds(1)) +
977 peer_max_ack_delay_;
978 return pto_delay * (1 << consecutive_pto_count_);
979}
980
wub967ba572019-04-01 09:27:52 -0700981QuicTime::Delta QuicSentPacketManager::GetSlowStartDuration() const {
982 if (send_algorithm_->GetCongestionControlType() != kBBR) {
983 return QuicTime::Delta::Infinite();
984 }
985
986 if (!send_algorithm_->InSlowStart()) {
987 return stats_->slowstart_duration;
988 }
989
990 return clock_->ApproximateNow() - stats_->slowstart_start_time +
991 stats_->slowstart_duration;
992}
993
vasilvvc48c8712019-03-11 13:38:16 -0700994std::string QuicSentPacketManager::GetDebugState() const {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500995 return send_algorithm_->GetDebugState();
996}
997
QUICHE teama6ef0a62019-03-07 20:34:33 -0500998void QuicSentPacketManager::SetSendAlgorithm(
999 CongestionControlType congestion_control_type) {
1000 SetSendAlgorithm(SendAlgorithmInterface::Create(
QUICHE team73957f12019-04-18 16:21:52 -07001001 clock_, &rtt_stats_, &unacked_packets_, congestion_control_type, random_,
1002 stats_, initial_congestion_window_));
QUICHE teama6ef0a62019-03-07 20:34:33 -05001003}
1004
1005void QuicSentPacketManager::SetSendAlgorithm(
1006 SendAlgorithmInterface* send_algorithm) {
1007 send_algorithm_.reset(send_algorithm);
1008 pacing_sender_.set_sender(send_algorithm);
1009}
1010
1011void QuicSentPacketManager::OnConnectionMigration(AddressChangeType type) {
1012 if (type == PORT_CHANGE || type == IPV4_SUBNET_CHANGE) {
1013 // Rtt and cwnd do not need to be reset when the peer address change is
1014 // considered to be caused by NATs.
1015 return;
1016 }
1017 consecutive_rto_count_ = 0;
1018 consecutive_tlp_count_ = 0;
fayangce0a3162019-08-15 09:05:36 -07001019 consecutive_pto_count_ = 0;
QUICHE teama6ef0a62019-03-07 20:34:33 -05001020 rtt_stats_.OnConnectionMigration();
1021 send_algorithm_->OnConnectionMigration();
1022}
1023
1024void QuicSentPacketManager::OnAckFrameStart(QuicPacketNumber largest_acked,
1025 QuicTime::Delta ack_delay_time,
1026 QuicTime ack_receive_time) {
1027 DCHECK(packets_acked_.empty());
1028 DCHECK_LE(largest_acked, unacked_packets_.largest_sent_packet());
1029 rtt_updated_ =
1030 MaybeUpdateRTT(largest_acked, ack_delay_time, ack_receive_time);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001031 last_ack_frame_.ack_delay_time = ack_delay_time;
1032 acked_packets_iter_ = last_ack_frame_.packets.rbegin();
1033}
1034
1035void QuicSentPacketManager::OnAckRange(QuicPacketNumber start,
1036 QuicPacketNumber end) {
1037 if (!last_ack_frame_.largest_acked.IsInitialized() ||
1038 end > last_ack_frame_.largest_acked + 1) {
1039 // Largest acked increases.
1040 unacked_packets_.IncreaseLargestAcked(end - 1);
1041 last_ack_frame_.largest_acked = end - 1;
1042 }
1043 // Drop ack ranges which ack packets below least_unacked.
1044 QuicPacketNumber least_unacked = unacked_packets_.GetLeastUnacked();
1045 if (least_unacked.IsInitialized() && end <= least_unacked) {
1046 return;
1047 }
1048 start = std::max(start, least_unacked);
1049 do {
1050 QuicPacketNumber newly_acked_start = start;
1051 if (acked_packets_iter_ != last_ack_frame_.packets.rend()) {
1052 newly_acked_start = std::max(start, acked_packets_iter_->max());
1053 }
1054 for (QuicPacketNumber acked = end - 1; acked >= newly_acked_start;
1055 --acked) {
1056 // Check if end is above the current range. If so add newly acked packets
1057 // in descending order.
1058 packets_acked_.push_back(AckedPacket(acked, 0, QuicTime::Zero()));
1059 if (acked == FirstSendingPacketNumber()) {
1060 break;
1061 }
1062 }
1063 if (acked_packets_iter_ == last_ack_frame_.packets.rend() ||
1064 start > acked_packets_iter_->min()) {
1065 // Finish adding all newly acked packets.
1066 return;
1067 }
1068 end = std::min(end, acked_packets_iter_->min());
1069 ++acked_packets_iter_;
1070 } while (start < end);
1071}
1072
1073void QuicSentPacketManager::OnAckTimestamp(QuicPacketNumber packet_number,
1074 QuicTime timestamp) {
1075 last_ack_frame_.received_packet_times.push_back({packet_number, timestamp});
1076 for (AckedPacket& packet : packets_acked_) {
1077 if (packet.packet_number == packet_number) {
1078 packet.receive_timestamp = timestamp;
1079 return;
1080 }
1081 }
1082}
1083
fayang3eb82212019-04-16 12:05:46 -07001084AckResult QuicSentPacketManager::OnAckFrameEnd(
1085 QuicTime ack_receive_time,
fayangf8e918b2019-07-16 13:03:16 -07001086 QuicPacketNumber ack_packet_number,
fayang3eb82212019-04-16 12:05:46 -07001087 EncryptionLevel ack_decrypted_level) {
QUICHE teama6ef0a62019-03-07 20:34:33 -05001088 QuicByteCount prior_bytes_in_flight = unacked_packets_.bytes_in_flight();
1089 // Reverse packets_acked_ so that it is in ascending order.
1090 reverse(packets_acked_.begin(), packets_acked_.end());
1091 for (AckedPacket& acked_packet : packets_acked_) {
1092 QuicTransmissionInfo* info =
1093 unacked_packets_.GetMutableTransmissionInfo(acked_packet.packet_number);
1094 if (!QuicUtils::IsAckable(info->state)) {
1095 if (info->state == ACKED) {
1096 QUIC_BUG << "Trying to ack an already acked packet: "
1097 << acked_packet.packet_number
1098 << ", last_ack_frame_: " << last_ack_frame_
1099 << ", least_unacked: " << unacked_packets_.GetLeastUnacked()
1100 << ", packets_acked_: " << packets_acked_;
1101 } else {
fayang3eb82212019-04-16 12:05:46 -07001102 QUIC_PEER_BUG << "Received "
dschinazief79a5f2019-10-04 10:32:54 -07001103 << EncryptionLevelToString(ack_decrypted_level)
fayang3eb82212019-04-16 12:05:46 -07001104 << " ack for unackable packet: "
QUICHE teama6ef0a62019-03-07 20:34:33 -05001105 << acked_packet.packet_number << " with state: "
1106 << QuicUtils::SentPacketStateToString(info->state);
fayang3eb82212019-04-16 12:05:46 -07001107 if (supports_multiple_packet_number_spaces()) {
1108 if (info->state == NEVER_SENT) {
1109 return UNSENT_PACKETS_ACKED;
1110 }
1111 return UNACKABLE_PACKETS_ACKED;
1112 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001113 }
1114 continue;
1115 }
fayang3eb82212019-04-16 12:05:46 -07001116 QUIC_DVLOG(1) << ENDPOINT << "Got an "
dschinazief79a5f2019-10-04 10:32:54 -07001117 << EncryptionLevelToString(ack_decrypted_level)
fayang19d2d5b2019-09-11 14:22:03 -07001118 << " ack for packet " << acked_packet.packet_number
1119 << " , state: "
1120 << QuicUtils::SentPacketStateToString(info->state);
fayang3eb82212019-04-16 12:05:46 -07001121 const PacketNumberSpace packet_number_space =
fayangbf3d2862019-06-20 14:13:44 -07001122 unacked_packets_.GetPacketNumberSpace(info->encryption_level);
fayang3eb82212019-04-16 12:05:46 -07001123 if (supports_multiple_packet_number_spaces() &&
1124 QuicUtils::GetPacketNumberSpace(ack_decrypted_level) !=
1125 packet_number_space) {
1126 return PACKETS_ACKED_IN_WRONG_PACKET_NUMBER_SPACE;
1127 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001128 last_ack_frame_.packets.Add(acked_packet.packet_number);
fayang70f9f932019-10-18 06:55:09 -07001129 if (info->encryption_level == ENCRYPTION_FORWARD_SECURE) {
1130 forward_secure_packet_acked_ = true;
1131 }
QUICHE teamc264e362019-03-19 14:21:06 -07001132 largest_packet_peer_knows_is_acked_.UpdateMax(info->largest_acked);
QUICHE teamc279cec2019-03-22 06:51:48 -07001133 if (supports_multiple_packet_number_spaces()) {
fayang3eb82212019-04-16 12:05:46 -07001134 largest_packets_peer_knows_is_acked_[packet_number_space].UpdateMax(
1135 info->largest_acked);
QUICHE teamc279cec2019-03-22 06:51:48 -07001136 }
QUICHE teama6ef0a62019-03-07 20:34:33 -05001137 // If data is associated with the most recent transmission of this
1138 // packet, then inform the caller.
1139 if (info->in_flight) {
1140 acked_packet.bytes_acked = info->bytes_sent;
1141 } else {
1142 // Unackable packets are skipped earlier.
1143 largest_newly_acked_ = acked_packet.packet_number;
1144 }
fayangbf3d2862019-06-20 14:13:44 -07001145 unacked_packets_.MaybeUpdateLargestAckedOfPacketNumberSpace(
1146 packet_number_space, acked_packet.packet_number);
fayang19d2d5b2019-09-11 14:22:03 -07001147 MarkPacketHandled(acked_packet.packet_number, info, ack_receive_time,
QUICHE team9467db02019-05-30 09:38:45 -07001148 last_ack_frame_.ack_delay_time,
1149 acked_packet.receive_timestamp);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001150 }
1151 const bool acked_new_packet = !packets_acked_.empty();
fayangf8e918b2019-07-16 13:03:16 -07001152 PostProcessNewlyAckedPackets(ack_packet_number, last_ack_frame_,
1153 ack_receive_time, rtt_updated_,
fayang3eb82212019-04-16 12:05:46 -07001154 prior_bytes_in_flight);
QUICHE teama6ef0a62019-03-07 20:34:33 -05001155
fayang3eb82212019-04-16 12:05:46 -07001156 return acked_new_packet ? PACKETS_NEWLY_ACKED : NO_PACKETS_NEWLY_ACKED;
QUICHE teama6ef0a62019-03-07 20:34:33 -05001157}
1158
1159void QuicSentPacketManager::SetDebugDelegate(DebugDelegate* debug_delegate) {
1160 debug_delegate_ = debug_delegate;
1161}
1162
1163void QuicSentPacketManager::OnApplicationLimited() {
1164 if (using_pacing_) {
1165 pacing_sender_.OnApplicationLimited();
1166 }
1167 send_algorithm_->OnApplicationLimited(unacked_packets_.bytes_in_flight());
1168 if (debug_delegate_ != nullptr) {
1169 debug_delegate_->OnApplicationLimited();
1170 }
1171}
1172
1173QuicTime QuicSentPacketManager::GetNextReleaseTime() const {
1174 return using_pacing_ ? pacing_sender_.ideal_next_packet_send_time()
1175 : QuicTime::Zero();
1176}
1177
1178void QuicSentPacketManager::SetInitialRtt(QuicTime::Delta rtt) {
1179 const QuicTime::Delta min_rtt =
1180 QuicTime::Delta::FromMicroseconds(kMinInitialRoundTripTimeUs);
1181 const QuicTime::Delta max_rtt =
1182 QuicTime::Delta::FromMicroseconds(kMaxInitialRoundTripTimeUs);
1183 rtt_stats_.set_initial_rtt(std::max(min_rtt, std::min(max_rtt, rtt)));
1184}
1185
QUICHE teamc279cec2019-03-22 06:51:48 -07001186void QuicSentPacketManager::EnableMultiplePacketNumberSpacesSupport() {
1187 unacked_packets_.EnableMultiplePacketNumberSpacesSupport();
1188}
1189
1190QuicPacketNumber QuicSentPacketManager::GetLargestAckedPacket(
1191 EncryptionLevel decrypted_packet_level) const {
1192 DCHECK(supports_multiple_packet_number_spaces());
1193 return unacked_packets_.GetLargestAckedOfPacketNumberSpace(
1194 QuicUtils::GetPacketNumberSpace(decrypted_packet_level));
1195}
1196
1197QuicPacketNumber QuicSentPacketManager::GetLargestSentPacket(
1198 EncryptionLevel decrypted_packet_level) const {
1199 DCHECK(supports_multiple_packet_number_spaces());
1200 return unacked_packets_.GetLargestSentPacketOfPacketNumberSpace(
1201 decrypted_packet_level);
1202}
1203
1204QuicPacketNumber QuicSentPacketManager::GetLargestPacketPeerKnowsIsAcked(
1205 EncryptionLevel decrypted_packet_level) const {
1206 DCHECK(supports_multiple_packet_number_spaces());
1207 return largest_packets_peer_knows_is_acked_[QuicUtils::GetPacketNumberSpace(
1208 decrypted_packet_level)];
1209}
1210
QUICHE teama6ef0a62019-03-07 20:34:33 -05001211#undef ENDPOINT // undef for jumbo builds
1212} // namespace quic