blob: ae1432a1956287cebc2cffbd96bef5133a7ce7ec [file] [log] [blame]
QUICHE teama6ef0a62019-03-07 20:34:33 -05001// Copyright 2016 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/congestion_control/bbr_sender.h"
6
7#include <algorithm>
8#include <sstream>
vasilvv872e7a32019-03-12 16:42:44 -07009#include <string>
QUICHE teama6ef0a62019-03-07 20:34:33 -050010
11#include "net/third_party/quiche/src/quic/core/congestion_control/rtt_stats.h"
12#include "net/third_party/quiche/src/quic/core/crypto/crypto_protocol.h"
wub967ba572019-04-01 09:27:52 -070013#include "net/third_party/quiche/src/quic/core/quic_time.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050014#include "net/third_party/quiche/src/quic/platform/api/quic_bug_tracker.h"
15#include "net/third_party/quiche/src/quic/platform/api/quic_fallthrough.h"
16#include "net/third_party/quiche/src/quic/platform/api/quic_flag_utils.h"
17#include "net/third_party/quiche/src/quic/platform/api/quic_flags.h"
18#include "net/third_party/quiche/src/quic/platform/api/quic_logging.h"
QUICHE teama6ef0a62019-03-07 20:34:33 -050019
20namespace quic {
21
22namespace {
23// Constants based on TCP defaults.
24// The minimum CWND to ensure delayed acks don't reduce bandwidth measurements.
25// Does not inflate the pacing rate.
26const QuicByteCount kDefaultMinimumCongestionWindow = 4 * kMaxSegmentSize;
27
28// The gain used for the STARTUP, equal to 2/ln(2).
29const float kDefaultHighGain = 2.885f;
30// The newly derived gain for STARTUP, equal to 4 * ln(2)
31const float kDerivedHighGain = 2.773f;
32// The newly derived CWND gain for STARTUP, 2.
wub5b352f12019-05-07 11:45:26 -070033const float kDerivedHighCWNDGain = 2.0f;
QUICHE teama6ef0a62019-03-07 20:34:33 -050034// The gain used in STARTUP after loss has been detected.
35// 1.5 is enough to allow for 25% exogenous loss and still observe a 25% growth
36// in measured bandwidth.
37const float kStartupAfterLossGain = 1.5f;
38// The cycle of gains used during the PROBE_BW stage.
39const float kPacingGain[] = {1.25, 0.75, 1, 1, 1, 1, 1, 1};
40
41// The length of the gain cycle.
42const size_t kGainCycleLength = sizeof(kPacingGain) / sizeof(kPacingGain[0]);
43// The size of the bandwidth filter window, in round-trips.
44const QuicRoundTripCount kBandwidthWindowSize = kGainCycleLength + 2;
45
46// The time after which the current min_rtt value expires.
47const QuicTime::Delta kMinRttExpiry = QuicTime::Delta::FromSeconds(10);
48// The minimum time the connection can spend in PROBE_RTT mode.
49const QuicTime::Delta kProbeRttTime = QuicTime::Delta::FromMilliseconds(200);
50// If the bandwidth does not increase by the factor of |kStartupGrowthTarget|
51// within |kRoundTripsWithoutGrowthBeforeExitingStartup| rounds, the connection
52// will exit the STARTUP mode.
53const float kStartupGrowthTarget = 1.25;
54const QuicRoundTripCount kRoundTripsWithoutGrowthBeforeExitingStartup = 3;
55// Coefficient of target congestion window to use when basing PROBE_RTT on BDP.
56const float kModerateProbeRttMultiplier = 0.75;
57// Coefficient to determine if a new RTT is sufficiently similar to min_rtt that
58// we don't need to enter PROBE_RTT.
59const float kSimilarMinRttThreshold = 1.125;
60
61} // namespace
62
63BbrSender::DebugState::DebugState(const BbrSender& sender)
64 : mode(sender.mode_),
65 max_bandwidth(sender.max_bandwidth_.GetBest()),
66 round_trip_count(sender.round_trip_count_),
67 gain_cycle_index(sender.cycle_current_offset_),
68 congestion_window(sender.congestion_window_),
69 is_at_full_bandwidth(sender.is_at_full_bandwidth_),
70 bandwidth_at_last_round(sender.bandwidth_at_last_round_),
71 rounds_without_bandwidth_gain(sender.rounds_without_bandwidth_gain_),
72 min_rtt(sender.min_rtt_),
73 min_rtt_timestamp(sender.min_rtt_timestamp_),
74 recovery_state(sender.recovery_state_),
75 recovery_window(sender.recovery_window_),
76 last_sample_is_app_limited(sender.last_sample_is_app_limited_),
77 end_of_app_limited_phase(sender.sampler_.end_of_app_limited_phase()) {}
78
79BbrSender::DebugState::DebugState(const DebugState& state) = default;
80
wub967ba572019-04-01 09:27:52 -070081BbrSender::BbrSender(QuicTime now,
82 const RttStats* rtt_stats,
QUICHE teama6ef0a62019-03-07 20:34:33 -050083 const QuicUnackedPacketMap* unacked_packets,
84 QuicPacketCount initial_tcp_congestion_window,
85 QuicPacketCount max_tcp_congestion_window,
wub967ba572019-04-01 09:27:52 -070086 QuicRandom* random,
87 QuicConnectionStats* stats)
QUICHE teama6ef0a62019-03-07 20:34:33 -050088 : rtt_stats_(rtt_stats),
89 unacked_packets_(unacked_packets),
90 random_(random),
wub967ba572019-04-01 09:27:52 -070091 stats_(stats),
QUICHE teama6ef0a62019-03-07 20:34:33 -050092 mode_(STARTUP),
vasilvv4abb5662019-08-14 08:23:49 -070093 sampler_(unacked_packets, kBandwidthWindowSize),
QUICHE teama6ef0a62019-03-07 20:34:33 -050094 round_trip_count_(0),
95 max_bandwidth_(kBandwidthWindowSize, QuicBandwidth::Zero(), 0),
96 max_ack_height_(kBandwidthWindowSize, 0, 0),
97 aggregation_epoch_start_time_(QuicTime::Zero()),
98 aggregation_epoch_bytes_(0),
99 min_rtt_(QuicTime::Delta::Zero()),
100 min_rtt_timestamp_(QuicTime::Zero()),
101 congestion_window_(initial_tcp_congestion_window * kDefaultTCPMSS),
102 initial_congestion_window_(initial_tcp_congestion_window *
103 kDefaultTCPMSS),
104 max_congestion_window_(max_tcp_congestion_window * kDefaultTCPMSS),
105 min_congestion_window_(kDefaultMinimumCongestionWindow),
106 high_gain_(kDefaultHighGain),
107 high_cwnd_gain_(kDefaultHighGain),
108 drain_gain_(1.f / kDefaultHighGain),
109 pacing_rate_(QuicBandwidth::Zero()),
110 pacing_gain_(1),
111 congestion_window_gain_(1),
112 congestion_window_gain_constant_(
danzh88e3e052019-06-13 11:47:18 -0700113 static_cast<float>(GetQuicFlag(FLAGS_quic_bbr_cwnd_gain))),
QUICHE teama6ef0a62019-03-07 20:34:33 -0500114 num_startup_rtts_(kRoundTripsWithoutGrowthBeforeExitingStartup),
115 exit_startup_on_loss_(false),
116 cycle_current_offset_(0),
117 last_cycle_start_(QuicTime::Zero()),
118 is_at_full_bandwidth_(false),
119 rounds_without_bandwidth_gain_(0),
120 bandwidth_at_last_round_(QuicBandwidth::Zero()),
121 exiting_quiescence_(false),
122 exit_probe_rtt_at_(QuicTime::Zero()),
123 probe_rtt_round_passed_(false),
124 last_sample_is_app_limited_(false),
125 has_non_app_limited_sample_(false),
126 flexible_app_limited_(false),
127 recovery_state_(NOT_IN_RECOVERY),
128 recovery_window_(max_congestion_window_),
QUICHE teama6ef0a62019-03-07 20:34:33 -0500129 slower_startup_(false),
130 rate_based_startup_(false),
131 startup_rate_reduction_multiplier_(0),
132 startup_bytes_lost_(0),
133 enable_ack_aggregation_during_startup_(false),
134 expire_ack_aggregation_in_startup_(false),
135 drain_to_target_(false),
136 probe_rtt_based_on_bdp_(false),
137 probe_rtt_skipped_if_similar_rtt_(false),
138 probe_rtt_disabled_if_app_limited_(false),
139 app_limited_since_last_probe_rtt_(false),
wub64c34112019-08-19 11:54:55 -0700140 min_rtt_since_last_probe_rtt_(QuicTime::Delta::Infinite()) {
wub967ba572019-04-01 09:27:52 -0700141 if (stats_) {
142 stats_->slowstart_count = 0;
143 stats_->slowstart_start_time = QuicTime::Zero();
144 }
145 EnterStartupMode(now);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500146}
147
148BbrSender::~BbrSender() {}
149
150void BbrSender::SetInitialCongestionWindowInPackets(
151 QuicPacketCount congestion_window) {
152 if (mode_ == STARTUP) {
153 initial_congestion_window_ = congestion_window * kDefaultTCPMSS;
154 congestion_window_ = congestion_window * kDefaultTCPMSS;
155 }
156}
157
158bool BbrSender::InSlowStart() const {
159 return mode_ == STARTUP;
160}
161
162void BbrSender::OnPacketSent(QuicTime sent_time,
163 QuicByteCount bytes_in_flight,
164 QuicPacketNumber packet_number,
165 QuicByteCount bytes,
166 HasRetransmittableData is_retransmittable) {
wub967ba572019-04-01 09:27:52 -0700167 if (stats_ && InSlowStart()) {
168 ++stats_->slowstart_packets_sent;
169 stats_->slowstart_bytes_sent += bytes;
170 }
171
QUICHE teama6ef0a62019-03-07 20:34:33 -0500172 last_sent_packet_ = packet_number;
173
174 if (bytes_in_flight == 0 && sampler_.is_app_limited()) {
175 exiting_quiescence_ = true;
176 }
177
wub64c34112019-08-19 11:54:55 -0700178 if (!sampler_.quic_track_ack_height_in_bandwidth_sampler()) {
179 if (!aggregation_epoch_start_time_.IsInitialized()) {
180 aggregation_epoch_start_time_ = sent_time;
181 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500182 }
183
184 sampler_.OnPacketSent(sent_time, packet_number, bytes, bytes_in_flight,
185 is_retransmittable);
186}
187
188bool BbrSender::CanSend(QuicByteCount bytes_in_flight) {
189 return bytes_in_flight < GetCongestionWindow();
190}
191
dschinazi17d42422019-06-18 16:35:07 -0700192QuicBandwidth BbrSender::PacingRate(QuicByteCount /*bytes_in_flight*/) const {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500193 if (pacing_rate_.IsZero()) {
194 return high_gain_ * QuicBandwidth::FromBytesAndTimeDelta(
195 initial_congestion_window_, GetMinRtt());
196 }
197 return pacing_rate_;
198}
199
200QuicBandwidth BbrSender::BandwidthEstimate() const {
201 return max_bandwidth_.GetBest();
202}
203
204QuicByteCount BbrSender::GetCongestionWindow() const {
205 if (mode_ == PROBE_RTT) {
206 return ProbeRttCongestionWindow();
207 }
208
209 if (InRecovery() && !(rate_based_startup_ && mode_ == STARTUP)) {
210 return std::min(congestion_window_, recovery_window_);
211 }
212
213 return congestion_window_;
214}
215
216QuicByteCount BbrSender::GetSlowStartThreshold() const {
217 return 0;
218}
219
220bool BbrSender::InRecovery() const {
221 return recovery_state_ != NOT_IN_RECOVERY;
222}
223
224bool BbrSender::ShouldSendProbingPacket() const {
225 if (pacing_gain_ <= 1) {
226 return false;
227 }
228
229 // TODO(b/77975811): If the pipe is highly under-utilized, consider not
230 // sending a probing transmission, because the extra bandwidth is not needed.
231 // If flexible_app_limited is enabled, check if the pipe is sufficiently full.
232 if (flexible_app_limited_) {
233 return !IsPipeSufficientlyFull();
234 } else {
235 return true;
236 }
237}
238
239bool BbrSender::IsPipeSufficientlyFull() const {
240 // See if we need more bytes in flight to see more bandwidth.
241 if (mode_ == STARTUP) {
242 // STARTUP exits if it doesn't observe a 25% bandwidth increase, so the CWND
243 // must be more than 25% above the target.
244 return unacked_packets_->bytes_in_flight() >=
245 GetTargetCongestionWindow(1.5);
246 }
247 if (pacing_gain_ > 1) {
248 // Super-unity PROBE_BW doesn't exit until 1.25 * BDP is achieved.
249 return unacked_packets_->bytes_in_flight() >=
250 GetTargetCongestionWindow(pacing_gain_);
251 }
252 // If bytes_in_flight are above the target congestion window, it should be
253 // possible to observe the same or more bandwidth if it's available.
254 return unacked_packets_->bytes_in_flight() >= GetTargetCongestionWindow(1.1);
255}
256
257void BbrSender::SetFromConfig(const QuicConfig& config,
258 Perspective perspective) {
259 if (config.HasClientRequestedIndependentOption(kLRTT, perspective)) {
260 exit_startup_on_loss_ = true;
261 }
262 if (config.HasClientRequestedIndependentOption(k1RTT, perspective)) {
263 num_startup_rtts_ = 1;
264 }
265 if (config.HasClientRequestedIndependentOption(k2RTT, perspective)) {
266 num_startup_rtts_ = 2;
267 }
268 if (config.HasClientRequestedIndependentOption(kBBRS, perspective)) {
269 slower_startup_ = true;
270 }
271 if (config.HasClientRequestedIndependentOption(kBBR3, perspective)) {
272 drain_to_target_ = true;
273 }
274 if (config.HasClientRequestedIndependentOption(kBBS1, perspective)) {
275 rate_based_startup_ = true;
276 }
277 if (GetQuicReloadableFlag(quic_bbr_startup_rate_reduction) &&
278 config.HasClientRequestedIndependentOption(kBBS4, perspective)) {
279 rate_based_startup_ = true;
280 // Hits 1.25x pacing multiplier when ~2/3 CWND is lost.
281 startup_rate_reduction_multiplier_ = 1;
282 }
283 if (GetQuicReloadableFlag(quic_bbr_startup_rate_reduction) &&
284 config.HasClientRequestedIndependentOption(kBBS5, perspective)) {
285 rate_based_startup_ = true;
286 // Hits 1.25x pacing multiplier when ~1/3 CWND is lost.
287 startup_rate_reduction_multiplier_ = 2;
288 }
289 if (config.HasClientRequestedIndependentOption(kBBR4, perspective)) {
wub64c34112019-08-19 11:54:55 -0700290 if (sampler_.quic_track_ack_height_in_bandwidth_sampler()) {
wubf35ea982019-08-06 16:19:38 -0700291 sampler_.SetMaxAckHeightTrackerWindowLength(2 * kBandwidthWindowSize);
292 } else {
293 max_ack_height_.SetWindowLength(2 * kBandwidthWindowSize);
294 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500295 }
296 if (config.HasClientRequestedIndependentOption(kBBR5, perspective)) {
wub64c34112019-08-19 11:54:55 -0700297 if (sampler_.quic_track_ack_height_in_bandwidth_sampler()) {
wubf35ea982019-08-06 16:19:38 -0700298 sampler_.SetMaxAckHeightTrackerWindowLength(4 * kBandwidthWindowSize);
299 } else {
300 max_ack_height_.SetWindowLength(4 * kBandwidthWindowSize);
301 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500302 }
303 if (GetQuicReloadableFlag(quic_bbr_less_probe_rtt) &&
304 config.HasClientRequestedIndependentOption(kBBR6, perspective)) {
305 QUIC_RELOADABLE_FLAG_COUNT_N(quic_bbr_less_probe_rtt, 1, 3);
306 probe_rtt_based_on_bdp_ = true;
307 }
308 if (GetQuicReloadableFlag(quic_bbr_less_probe_rtt) &&
309 config.HasClientRequestedIndependentOption(kBBR7, perspective)) {
310 QUIC_RELOADABLE_FLAG_COUNT_N(quic_bbr_less_probe_rtt, 2, 3);
311 probe_rtt_skipped_if_similar_rtt_ = true;
312 }
313 if (GetQuicReloadableFlag(quic_bbr_less_probe_rtt) &&
314 config.HasClientRequestedIndependentOption(kBBR8, perspective)) {
315 QUIC_RELOADABLE_FLAG_COUNT_N(quic_bbr_less_probe_rtt, 3, 3);
316 probe_rtt_disabled_if_app_limited_ = true;
317 }
318 if (GetQuicReloadableFlag(quic_bbr_flexible_app_limited) &&
319 config.HasClientRequestedIndependentOption(kBBR9, perspective)) {
320 QUIC_RELOADABLE_FLAG_COUNT(quic_bbr_flexible_app_limited);
321 flexible_app_limited_ = true;
322 }
ianswettcfef5c92019-06-18 07:58:39 -0700323 if (config.HasClientRequestedIndependentOption(kBBQ1, perspective)) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500324 set_high_gain(kDerivedHighGain);
325 set_high_cwnd_gain(kDerivedHighGain);
326 set_drain_gain(1.f / kDerivedHighGain);
327 }
ianswettcfef5c92019-06-18 07:58:39 -0700328 if (config.HasClientRequestedIndependentOption(kBBQ2, perspective)) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500329 set_high_cwnd_gain(kDerivedHighCWNDGain);
330 }
ianswettcfef5c92019-06-18 07:58:39 -0700331 if (config.HasClientRequestedIndependentOption(kBBQ3, perspective)) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500332 enable_ack_aggregation_during_startup_ = true;
333 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500334 if (GetQuicReloadableFlag(quic_bbr_slower_startup4) &&
335 config.HasClientRequestedIndependentOption(kBBQ5, perspective)) {
336 QUIC_RELOADABLE_FLAG_COUNT(quic_bbr_slower_startup4);
337 expire_ack_aggregation_in_startup_ = true;
338 }
339 if (config.HasClientRequestedIndependentOption(kMIN1, perspective)) {
340 min_congestion_window_ = kMaxSegmentSize;
341 }
342}
343
344void BbrSender::AdjustNetworkParameters(QuicBandwidth bandwidth,
fayangf1b99dc2019-05-14 06:29:18 -0700345 QuicTime::Delta rtt,
346 bool allow_cwnd_to_decrease) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500347 if (!bandwidth.IsZero()) {
348 max_bandwidth_.Update(bandwidth, round_trip_count_);
349 }
350 if (!rtt.IsZero() && (min_rtt_ > rtt || min_rtt_.IsZero())) {
351 min_rtt_ = rtt;
352 }
fayangbe83ecd2019-04-26 13:58:09 -0700353 if (GetQuicReloadableFlag(quic_fix_bbr_cwnd_in_bandwidth_resumption) &&
354 mode_ == STARTUP) {
fayangc050d7a2019-05-17 11:20:38 -0700355 if (bandwidth.IsZero()) {
356 // Ignore bad bandwidth samples.
357 QUIC_RELOADABLE_FLAG_COUNT_N(quic_fix_bbr_cwnd_in_bandwidth_resumption, 3,
358 3);
359 return;
360 }
fayangbe83ecd2019-04-26 13:58:09 -0700361 const QuicByteCount new_cwnd =
fayangf1b99dc2019-05-14 06:29:18 -0700362 std::max(kMinInitialCongestionWindow * kDefaultTCPMSS,
363 std::min(kMaxInitialCongestionWindow * kDefaultTCPMSS,
364 bandwidth * rtt_stats_->SmoothedOrInitialRtt()));
fayang14650e42019-06-04 10:30:33 -0700365 if (!rtt_stats_->smoothed_rtt().IsZero()) {
366 QUIC_CODE_COUNT(quic_smoothed_rtt_available);
367 } else if (rtt_stats_->initial_rtt() !=
368 QuicTime::Delta::FromMilliseconds(kInitialRttMs)) {
369 QUIC_CODE_COUNT(quic_client_initial_rtt_available);
370 } else {
371 QUIC_CODE_COUNT(quic_default_initial_rtt);
372 }
fayangbe83ecd2019-04-26 13:58:09 -0700373 if (new_cwnd > congestion_window_) {
fayanged966432019-04-29 06:50:05 -0700374 QUIC_RELOADABLE_FLAG_COUNT_N(quic_fix_bbr_cwnd_in_bandwidth_resumption, 1,
fayangc050d7a2019-05-17 11:20:38 -0700375 3);
fayanged966432019-04-29 06:50:05 -0700376 } else {
377 QUIC_RELOADABLE_FLAG_COUNT_N(quic_fix_bbr_cwnd_in_bandwidth_resumption, 2,
fayangc050d7a2019-05-17 11:20:38 -0700378 3);
fayangbe83ecd2019-04-26 13:58:09 -0700379 }
fayangc050d7a2019-05-17 11:20:38 -0700380 if (new_cwnd < congestion_window_ && !allow_cwnd_to_decrease) {
381 // Only decrease cwnd if allow_cwnd_to_decrease is true.
fayangf1b99dc2019-05-14 06:29:18 -0700382 return;
383 }
fayang8cafde02019-06-04 06:21:04 -0700384 if (GetQuicReloadableFlag(quic_conservative_cwnd_and_pacing_gains)) {
385 // Decreases cwnd gain and pacing gain. Please note, if pacing_rate_ has
386 // been calculated, it cannot decrease in STARTUP phase.
387 QUIC_RELOADABLE_FLAG_COUNT(quic_conservative_cwnd_and_pacing_gains);
388 set_high_gain(kDerivedHighCWNDGain);
389 set_high_cwnd_gain(kDerivedHighCWNDGain);
390 }
fayangf1b99dc2019-05-14 06:29:18 -0700391 congestion_window_ = new_cwnd;
fayangbe83ecd2019-04-26 13:58:09 -0700392 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500393}
394
395void BbrSender::OnCongestionEvent(bool /*rtt_updated*/,
396 QuicByteCount prior_in_flight,
397 QuicTime event_time,
398 const AckedPacketVector& acked_packets,
399 const LostPacketVector& lost_packets) {
400 const QuicByteCount total_bytes_acked_before = sampler_.total_bytes_acked();
401
402 bool is_round_start = false;
403 bool min_rtt_expired = false;
404
405 DiscardLostPackets(lost_packets);
406
407 // Input the new data into the BBR model of the connection.
408 QuicByteCount excess_acked = 0;
409 if (!acked_packets.empty()) {
410 QuicPacketNumber last_acked_packet = acked_packets.rbegin()->packet_number;
411 is_round_start = UpdateRoundTripCounter(last_acked_packet);
412 min_rtt_expired = UpdateBandwidthAndMinRtt(event_time, acked_packets);
413 UpdateRecoveryState(last_acked_packet, !lost_packets.empty(),
414 is_round_start);
415
416 const QuicByteCount bytes_acked =
417 sampler_.total_bytes_acked() - total_bytes_acked_before;
418
wub64c34112019-08-19 11:54:55 -0700419 excess_acked = sampler_.quic_track_ack_height_in_bandwidth_sampler()
wubf35ea982019-08-06 16:19:38 -0700420 ? sampler_.OnAckEventEnd(max_bandwidth_.GetBest(),
421 round_trip_count_)
422 : UpdateAckAggregationBytes(event_time, bytes_acked);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500423 }
424
425 // Handle logic specific to PROBE_BW mode.
426 if (mode_ == PROBE_BW) {
427 UpdateGainCyclePhase(event_time, prior_in_flight, !lost_packets.empty());
428 }
429
430 // Handle logic specific to STARTUP and DRAIN modes.
431 if (is_round_start && !is_at_full_bandwidth_) {
432 CheckIfFullBandwidthReached();
433 }
434 MaybeExitStartupOrDrain(event_time);
435
436 // Handle logic specific to PROBE_RTT.
437 MaybeEnterOrExitProbeRtt(event_time, is_round_start, min_rtt_expired);
438
439 // Calculate number of packets acked and lost.
440 QuicByteCount bytes_acked =
441 sampler_.total_bytes_acked() - total_bytes_acked_before;
442 QuicByteCount bytes_lost = 0;
443 for (const auto& packet : lost_packets) {
444 bytes_lost += packet.bytes_lost;
445 }
446
447 // After the model is updated, recalculate the pacing rate and congestion
448 // window.
449 CalculatePacingRate();
450 CalculateCongestionWindow(bytes_acked, excess_acked);
451 CalculateRecoveryWindow(bytes_acked, bytes_lost);
452
453 // Cleanup internal state.
454 sampler_.RemoveObsoletePackets(unacked_packets_->GetLeastUnacked());
455}
456
457CongestionControlType BbrSender::GetCongestionControlType() const {
458 return kBBR;
459}
460
461QuicTime::Delta BbrSender::GetMinRtt() const {
462 return !min_rtt_.IsZero() ? min_rtt_ : rtt_stats_->initial_rtt();
463}
464
465QuicByteCount BbrSender::GetTargetCongestionWindow(float gain) const {
466 QuicByteCount bdp = GetMinRtt() * BandwidthEstimate();
467 QuicByteCount congestion_window = gain * bdp;
468
469 // BDP estimate will be zero if no bandwidth samples are available yet.
470 if (congestion_window == 0) {
471 congestion_window = gain * initial_congestion_window_;
472 }
473
474 return std::max(congestion_window, min_congestion_window_);
475}
476
477QuicByteCount BbrSender::ProbeRttCongestionWindow() const {
478 if (probe_rtt_based_on_bdp_) {
479 return GetTargetCongestionWindow(kModerateProbeRttMultiplier);
480 }
481 return min_congestion_window_;
482}
483
wub967ba572019-04-01 09:27:52 -0700484void BbrSender::EnterStartupMode(QuicTime now) {
485 if (stats_) {
486 ++stats_->slowstart_count;
487 DCHECK_EQ(stats_->slowstart_start_time, QuicTime::Zero()) << mode_;
488 stats_->slowstart_start_time = now;
489 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500490 mode_ = STARTUP;
491 pacing_gain_ = high_gain_;
492 congestion_window_gain_ = high_cwnd_gain_;
493}
494
495void BbrSender::EnterProbeBandwidthMode(QuicTime now) {
496 mode_ = PROBE_BW;
497 congestion_window_gain_ = congestion_window_gain_constant_;
498
499 // Pick a random offset for the gain cycle out of {0, 2..7} range. 1 is
500 // excluded because in that case increased gain and decreased gain would not
501 // follow each other.
502 cycle_current_offset_ = random_->RandUint64() % (kGainCycleLength - 1);
503 if (cycle_current_offset_ >= 1) {
504 cycle_current_offset_ += 1;
505 }
506
507 last_cycle_start_ = now;
508 pacing_gain_ = kPacingGain[cycle_current_offset_];
509}
510
511void BbrSender::DiscardLostPackets(const LostPacketVector& lost_packets) {
512 for (const LostPacket& packet : lost_packets) {
513 sampler_.OnPacketLost(packet.packet_number);
wub967ba572019-04-01 09:27:52 -0700514 if (mode_ == STARTUP) {
515 if (stats_) {
516 ++stats_->slowstart_packets_lost;
517 stats_->slowstart_bytes_lost += packet.bytes_lost;
518 }
519 if (startup_rate_reduction_multiplier_ != 0) {
520 startup_bytes_lost_ += packet.bytes_lost;
521 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500522 }
523 }
524}
525
526bool BbrSender::UpdateRoundTripCounter(QuicPacketNumber last_acked_packet) {
527 if (!current_round_trip_end_.IsInitialized() ||
528 last_acked_packet > current_round_trip_end_) {
529 round_trip_count_++;
530 current_round_trip_end_ = last_sent_packet_;
wub967ba572019-04-01 09:27:52 -0700531 if (stats_ && InSlowStart()) {
532 ++stats_->slowstart_num_rtts;
533 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500534 return true;
535 }
536
537 return false;
538}
539
540bool BbrSender::UpdateBandwidthAndMinRtt(
541 QuicTime now,
542 const AckedPacketVector& acked_packets) {
543 QuicTime::Delta sample_min_rtt = QuicTime::Delta::Infinite();
544 for (const auto& packet : acked_packets) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500545 BandwidthSample bandwidth_sample =
546 sampler_.OnPacketAcknowledged(now, packet.packet_number);
wub03637f52019-05-30 10:52:20 -0700547 if (!bandwidth_sample.state_at_send.is_valid) {
wubfb4e2132019-04-05 12:30:09 -0700548 // From the sampler's perspective, the packet has never been sent, or the
549 // packet has been acked or marked as lost previously.
550 continue;
551 }
552
wub254545c2019-04-04 13:56:52 -0700553 last_sample_is_app_limited_ = bandwidth_sample.state_at_send.is_app_limited;
554 has_non_app_limited_sample_ |=
555 !bandwidth_sample.state_at_send.is_app_limited;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500556 if (!bandwidth_sample.rtt.IsZero()) {
557 sample_min_rtt = std::min(sample_min_rtt, bandwidth_sample.rtt);
558 }
559
wub254545c2019-04-04 13:56:52 -0700560 if (!bandwidth_sample.state_at_send.is_app_limited ||
QUICHE teama6ef0a62019-03-07 20:34:33 -0500561 bandwidth_sample.bandwidth > BandwidthEstimate()) {
562 max_bandwidth_.Update(bandwidth_sample.bandwidth, round_trip_count_);
563 }
564 }
565
566 // If none of the RTT samples are valid, return immediately.
567 if (sample_min_rtt.IsInfinite()) {
568 return false;
569 }
570 min_rtt_since_last_probe_rtt_ =
571 std::min(min_rtt_since_last_probe_rtt_, sample_min_rtt);
572
573 // Do not expire min_rtt if none was ever available.
574 bool min_rtt_expired =
575 !min_rtt_.IsZero() && (now > (min_rtt_timestamp_ + kMinRttExpiry));
576
577 if (min_rtt_expired || sample_min_rtt < min_rtt_ || min_rtt_.IsZero()) {
578 QUIC_DVLOG(2) << "Min RTT updated, old value: " << min_rtt_
579 << ", new value: " << sample_min_rtt
580 << ", current time: " << now.ToDebuggingValue();
581
582 if (min_rtt_expired && ShouldExtendMinRttExpiry()) {
583 min_rtt_expired = false;
584 } else {
585 min_rtt_ = sample_min_rtt;
586 }
587 min_rtt_timestamp_ = now;
588 // Reset since_last_probe_rtt fields.
589 min_rtt_since_last_probe_rtt_ = QuicTime::Delta::Infinite();
590 app_limited_since_last_probe_rtt_ = false;
591 }
592 DCHECK(!min_rtt_.IsZero());
593
594 return min_rtt_expired;
595}
596
597bool BbrSender::ShouldExtendMinRttExpiry() const {
598 if (probe_rtt_disabled_if_app_limited_ && app_limited_since_last_probe_rtt_) {
599 // Extend the current min_rtt if we've been app limited recently.
600 return true;
601 }
602 const bool min_rtt_increased_since_last_probe =
603 min_rtt_since_last_probe_rtt_ > min_rtt_ * kSimilarMinRttThreshold;
604 if (probe_rtt_skipped_if_similar_rtt_ && app_limited_since_last_probe_rtt_ &&
605 !min_rtt_increased_since_last_probe) {
606 // Extend the current min_rtt if we've been app limited recently and an rtt
607 // has been measured in that time that's less than 12.5% more than the
608 // current min_rtt.
609 return true;
610 }
611 return false;
612}
613
614void BbrSender::UpdateGainCyclePhase(QuicTime now,
615 QuicByteCount prior_in_flight,
616 bool has_losses) {
617 const QuicByteCount bytes_in_flight = unacked_packets_->bytes_in_flight();
618 // In most cases, the cycle is advanced after an RTT passes.
619 bool should_advance_gain_cycling = now - last_cycle_start_ > GetMinRtt();
620
621 // If the pacing gain is above 1.0, the connection is trying to probe the
622 // bandwidth by increasing the number of bytes in flight to at least
623 // pacing_gain * BDP. Make sure that it actually reaches the target, as long
624 // as there are no losses suggesting that the buffers are not able to hold
625 // that much.
626 if (pacing_gain_ > 1.0 && !has_losses &&
627 prior_in_flight < GetTargetCongestionWindow(pacing_gain_)) {
628 should_advance_gain_cycling = false;
629 }
630
631 // If pacing gain is below 1.0, the connection is trying to drain the extra
632 // queue which could have been incurred by probing prior to it. If the number
633 // of bytes in flight falls down to the estimated BDP value earlier, conclude
634 // that the queue has been successfully drained and exit this cycle early.
635 if (pacing_gain_ < 1.0 && bytes_in_flight <= GetTargetCongestionWindow(1)) {
636 should_advance_gain_cycling = true;
637 }
638
639 if (should_advance_gain_cycling) {
640 cycle_current_offset_ = (cycle_current_offset_ + 1) % kGainCycleLength;
641 last_cycle_start_ = now;
642 // Stay in low gain mode until the target BDP is hit.
643 // Low gain mode will be exited immediately when the target BDP is achieved.
644 if (drain_to_target_ && pacing_gain_ < 1 &&
645 kPacingGain[cycle_current_offset_] == 1 &&
646 bytes_in_flight > GetTargetCongestionWindow(1)) {
647 return;
648 }
649 pacing_gain_ = kPacingGain[cycle_current_offset_];
650 }
651}
652
653void BbrSender::CheckIfFullBandwidthReached() {
654 if (last_sample_is_app_limited_) {
655 return;
656 }
657
658 QuicBandwidth target = bandwidth_at_last_round_ * kStartupGrowthTarget;
659 if (BandwidthEstimate() >= target) {
660 bandwidth_at_last_round_ = BandwidthEstimate();
661 rounds_without_bandwidth_gain_ = 0;
662 if (expire_ack_aggregation_in_startup_) {
663 // Expire old excess delivery measurements now that bandwidth increased.
wub64c34112019-08-19 11:54:55 -0700664 if (sampler_.quic_track_ack_height_in_bandwidth_sampler()) {
wubf35ea982019-08-06 16:19:38 -0700665 sampler_.ResetMaxAckHeightTracker(0, round_trip_count_);
666 } else {
667 max_ack_height_.Reset(0, round_trip_count_);
668 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500669 }
670 return;
671 }
672
673 rounds_without_bandwidth_gain_++;
674 if ((rounds_without_bandwidth_gain_ >= num_startup_rtts_) ||
675 (exit_startup_on_loss_ && InRecovery())) {
676 DCHECK(has_non_app_limited_sample_);
677 is_at_full_bandwidth_ = true;
678 }
679}
680
681void BbrSender::MaybeExitStartupOrDrain(QuicTime now) {
682 if (mode_ == STARTUP && is_at_full_bandwidth_) {
wub967ba572019-04-01 09:27:52 -0700683 OnExitStartup(now);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500684 mode_ = DRAIN;
685 pacing_gain_ = drain_gain_;
686 congestion_window_gain_ = high_cwnd_gain_;
687 }
688 if (mode_ == DRAIN &&
689 unacked_packets_->bytes_in_flight() <= GetTargetCongestionWindow(1)) {
690 EnterProbeBandwidthMode(now);
691 }
692}
693
wub967ba572019-04-01 09:27:52 -0700694void BbrSender::OnExitStartup(QuicTime now) {
695 DCHECK_EQ(mode_, STARTUP);
696 if (stats_) {
697 DCHECK_NE(stats_->slowstart_start_time, QuicTime::Zero());
698 if (now > stats_->slowstart_start_time) {
699 stats_->slowstart_duration =
700 now - stats_->slowstart_start_time + stats_->slowstart_duration;
701 }
702 stats_->slowstart_start_time = QuicTime::Zero();
703 }
704}
705
QUICHE teama6ef0a62019-03-07 20:34:33 -0500706void BbrSender::MaybeEnterOrExitProbeRtt(QuicTime now,
707 bool is_round_start,
708 bool min_rtt_expired) {
709 if (min_rtt_expired && !exiting_quiescence_ && mode_ != PROBE_RTT) {
wub967ba572019-04-01 09:27:52 -0700710 if (InSlowStart()) {
711 OnExitStartup(now);
712 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500713 mode_ = PROBE_RTT;
714 pacing_gain_ = 1;
715 // Do not decide on the time to exit PROBE_RTT until the |bytes_in_flight|
716 // is at the target small value.
717 exit_probe_rtt_at_ = QuicTime::Zero();
718 }
719
720 if (mode_ == PROBE_RTT) {
721 sampler_.OnAppLimited();
722
723 if (exit_probe_rtt_at_ == QuicTime::Zero()) {
724 // If the window has reached the appropriate size, schedule exiting
725 // PROBE_RTT. The CWND during PROBE_RTT is kMinimumCongestionWindow, but
726 // we allow an extra packet since QUIC checks CWND before sending a
727 // packet.
728 if (unacked_packets_->bytes_in_flight() <
dschinazi66dea072019-04-09 11:41:06 -0700729 ProbeRttCongestionWindow() + kMaxOutgoingPacketSize) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500730 exit_probe_rtt_at_ = now + kProbeRttTime;
731 probe_rtt_round_passed_ = false;
732 }
733 } else {
734 if (is_round_start) {
735 probe_rtt_round_passed_ = true;
736 }
737 if (now >= exit_probe_rtt_at_ && probe_rtt_round_passed_) {
738 min_rtt_timestamp_ = now;
739 if (!is_at_full_bandwidth_) {
wub967ba572019-04-01 09:27:52 -0700740 EnterStartupMode(now);
QUICHE teama6ef0a62019-03-07 20:34:33 -0500741 } else {
742 EnterProbeBandwidthMode(now);
743 }
744 }
745 }
746 }
747
748 exiting_quiescence_ = false;
749}
750
751void BbrSender::UpdateRecoveryState(QuicPacketNumber last_acked_packet,
752 bool has_losses,
753 bool is_round_start) {
754 // Exit recovery when there are no losses for a round.
755 if (has_losses) {
756 end_recovery_at_ = last_sent_packet_;
757 }
758
759 switch (recovery_state_) {
760 case NOT_IN_RECOVERY:
761 // Enter conservation on the first loss.
762 if (has_losses) {
763 recovery_state_ = CONSERVATION;
764 // This will cause the |recovery_window_| to be set to the correct
765 // value in CalculateRecoveryWindow().
766 recovery_window_ = 0;
767 // Since the conservation phase is meant to be lasting for a whole
768 // round, extend the current round as if it were started right now.
769 current_round_trip_end_ = last_sent_packet_;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500770 }
771 break;
772
773 case CONSERVATION:
774 if (is_round_start) {
775 recovery_state_ = GROWTH;
776 }
777 QUIC_FALLTHROUGH_INTENDED;
778
779 case GROWTH:
780 // Exit recovery if appropriate.
781 if (!has_losses && last_acked_packet > end_recovery_at_) {
782 recovery_state_ = NOT_IN_RECOVERY;
QUICHE teama6ef0a62019-03-07 20:34:33 -0500783 }
784
785 break;
786 }
QUICHE teama6ef0a62019-03-07 20:34:33 -0500787}
788
789// TODO(ianswett): Move this logic into BandwidthSampler.
790QuicByteCount BbrSender::UpdateAckAggregationBytes(
791 QuicTime ack_time,
792 QuicByteCount newly_acked_bytes) {
793 // Compute how many bytes are expected to be delivered, assuming max bandwidth
794 // is correct.
795 QuicByteCount expected_bytes_acked =
796 max_bandwidth_.GetBest() * (ack_time - aggregation_epoch_start_time_);
797 // Reset the current aggregation epoch as soon as the ack arrival rate is less
798 // than or equal to the max bandwidth.
799 if (aggregation_epoch_bytes_ <= expected_bytes_acked) {
800 // Reset to start measuring a new aggregation epoch.
801 aggregation_epoch_bytes_ = newly_acked_bytes;
802 aggregation_epoch_start_time_ = ack_time;
803 return 0;
804 }
805
806 // Compute how many extra bytes were delivered vs max bandwidth.
807 // Include the bytes most recently acknowledged to account for stretch acks.
808 aggregation_epoch_bytes_ += newly_acked_bytes;
809 max_ack_height_.Update(aggregation_epoch_bytes_ - expected_bytes_acked,
810 round_trip_count_);
811 return aggregation_epoch_bytes_ - expected_bytes_acked;
812}
813
814void BbrSender::CalculatePacingRate() {
815 if (BandwidthEstimate().IsZero()) {
816 return;
817 }
818
819 QuicBandwidth target_rate = pacing_gain_ * BandwidthEstimate();
820 if (is_at_full_bandwidth_) {
821 pacing_rate_ = target_rate;
822 return;
823 }
824
825 // Pace at the rate of initial_window / RTT as soon as RTT measurements are
826 // available.
827 if (pacing_rate_.IsZero() && !rtt_stats_->min_rtt().IsZero()) {
828 pacing_rate_ = QuicBandwidth::FromBytesAndTimeDelta(
829 initial_congestion_window_, rtt_stats_->min_rtt());
830 return;
831 }
832 // Slow the pacing rate in STARTUP once loss has ever been detected.
833 const bool has_ever_detected_loss = end_recovery_at_.IsInitialized();
834 if (slower_startup_ && has_ever_detected_loss &&
835 has_non_app_limited_sample_) {
836 pacing_rate_ = kStartupAfterLossGain * BandwidthEstimate();
837 return;
838 }
839
840 // Slow the pacing rate in STARTUP by the bytes_lost / CWND.
841 if (startup_rate_reduction_multiplier_ != 0 && has_ever_detected_loss &&
842 has_non_app_limited_sample_) {
843 pacing_rate_ =
844 (1 - (startup_bytes_lost_ * startup_rate_reduction_multiplier_ * 1.0f /
845 congestion_window_)) *
846 target_rate;
847 // Ensure the pacing rate doesn't drop below the startup growth target times
848 // the bandwidth estimate.
849 pacing_rate_ =
850 std::max(pacing_rate_, kStartupGrowthTarget * BandwidthEstimate());
851 return;
852 }
853
854 // Do not decrease the pacing rate during startup.
855 pacing_rate_ = std::max(pacing_rate_, target_rate);
856}
857
858void BbrSender::CalculateCongestionWindow(QuicByteCount bytes_acked,
859 QuicByteCount excess_acked) {
860 if (mode_ == PROBE_RTT) {
861 return;
862 }
863
864 QuicByteCount target_window =
865 GetTargetCongestionWindow(congestion_window_gain_);
866 if (is_at_full_bandwidth_) {
867 // Add the max recently measured ack aggregation to CWND.
wub64c34112019-08-19 11:54:55 -0700868 target_window += sampler_.quic_track_ack_height_in_bandwidth_sampler()
wubf35ea982019-08-06 16:19:38 -0700869 ? sampler_.max_ack_height()
870 : max_ack_height_.GetBest();
QUICHE teama6ef0a62019-03-07 20:34:33 -0500871 } else if (enable_ack_aggregation_during_startup_) {
872 // Add the most recent excess acked. Because CWND never decreases in
873 // STARTUP, this will automatically create a very localized max filter.
874 target_window += excess_acked;
875 }
876
877 // Instead of immediately setting the target CWND as the new one, BBR grows
878 // the CWND towards |target_window| by only increasing it |bytes_acked| at a
879 // time.
880 const bool add_bytes_acked =
881 !GetQuicReloadableFlag(quic_bbr_no_bytes_acked_in_startup_recovery) ||
882 !InRecovery();
883 if (is_at_full_bandwidth_) {
884 congestion_window_ =
885 std::min(target_window, congestion_window_ + bytes_acked);
886 } else if (add_bytes_acked &&
887 (congestion_window_ < target_window ||
888 sampler_.total_bytes_acked() < initial_congestion_window_)) {
889 // If the connection is not yet out of startup phase, do not decrease the
890 // window.
891 congestion_window_ = congestion_window_ + bytes_acked;
892 }
893
894 // Enforce the limits on the congestion window.
895 congestion_window_ = std::max(congestion_window_, min_congestion_window_);
896 congestion_window_ = std::min(congestion_window_, max_congestion_window_);
897}
898
899void BbrSender::CalculateRecoveryWindow(QuicByteCount bytes_acked,
900 QuicByteCount bytes_lost) {
901 if (rate_based_startup_ && mode_ == STARTUP) {
902 return;
903 }
904
905 if (recovery_state_ == NOT_IN_RECOVERY) {
906 return;
907 }
908
909 // Set up the initial recovery window.
910 if (recovery_window_ == 0) {
911 recovery_window_ = unacked_packets_->bytes_in_flight() + bytes_acked;
912 recovery_window_ = std::max(min_congestion_window_, recovery_window_);
913 return;
914 }
915
916 // Remove losses from the recovery window, while accounting for a potential
917 // integer underflow.
918 recovery_window_ = recovery_window_ >= bytes_lost
919 ? recovery_window_ - bytes_lost
920 : kMaxSegmentSize;
921
922 // In CONSERVATION mode, just subtracting losses is sufficient. In GROWTH,
923 // release additional |bytes_acked| to achieve a slow-start-like behavior.
924 if (recovery_state_ == GROWTH) {
925 recovery_window_ += bytes_acked;
926 }
927
928 // Sanity checks. Ensure that we always allow to send at least an MSS or
929 // |bytes_acked| in response, whichever is larger.
930 recovery_window_ = std::max(
931 recovery_window_, unacked_packets_->bytes_in_flight() + bytes_acked);
932 if (GetQuicReloadableFlag(quic_bbr_one_mss_conservation)) {
933 recovery_window_ =
934 std::max(recovery_window_,
935 unacked_packets_->bytes_in_flight() + kMaxSegmentSize);
936 }
937 recovery_window_ = std::max(min_congestion_window_, recovery_window_);
938}
939
vasilvvc48c8712019-03-11 13:38:16 -0700940std::string BbrSender::GetDebugState() const {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500941 std::ostringstream stream;
942 stream << ExportDebugState();
943 return stream.str();
944}
945
946void BbrSender::OnApplicationLimited(QuicByteCount bytes_in_flight) {
947 if (bytes_in_flight >= GetCongestionWindow()) {
948 return;
949 }
950 if (flexible_app_limited_ && IsPipeSufficientlyFull()) {
951 return;
952 }
953
954 app_limited_since_last_probe_rtt_ = true;
955 sampler_.OnAppLimited();
956 QUIC_DVLOG(2) << "Becoming application limited. Last sent packet: "
957 << last_sent_packet_ << ", CWND: " << GetCongestionWindow();
958}
959
960BbrSender::DebugState BbrSender::ExportDebugState() const {
961 return DebugState(*this);
962}
963
vasilvvc48c8712019-03-11 13:38:16 -0700964static std::string ModeToString(BbrSender::Mode mode) {
QUICHE teama6ef0a62019-03-07 20:34:33 -0500965 switch (mode) {
966 case BbrSender::STARTUP:
967 return "STARTUP";
968 case BbrSender::DRAIN:
969 return "DRAIN";
970 case BbrSender::PROBE_BW:
971 return "PROBE_BW";
972 case BbrSender::PROBE_RTT:
973 return "PROBE_RTT";
974 }
975 return "???";
976}
977
978std::ostream& operator<<(std::ostream& os, const BbrSender::Mode& mode) {
979 os << ModeToString(mode);
980 return os;
981}
982
983std::ostream& operator<<(std::ostream& os, const BbrSender::DebugState& state) {
984 os << "Mode: " << ModeToString(state.mode) << std::endl;
985 os << "Maximum bandwidth: " << state.max_bandwidth << std::endl;
986 os << "Round trip counter: " << state.round_trip_count << std::endl;
987 os << "Gain cycle index: " << static_cast<int>(state.gain_cycle_index)
988 << std::endl;
989 os << "Congestion window: " << state.congestion_window << " bytes"
990 << std::endl;
991
992 if (state.mode == BbrSender::STARTUP) {
993 os << "(startup) Bandwidth at last round: " << state.bandwidth_at_last_round
994 << std::endl;
995 os << "(startup) Rounds without gain: "
996 << state.rounds_without_bandwidth_gain << std::endl;
997 }
998
999 os << "Minimum RTT: " << state.min_rtt << std::endl;
1000 os << "Minimum RTT timestamp: " << state.min_rtt_timestamp.ToDebuggingValue()
1001 << std::endl;
1002
1003 os << "Last sample is app-limited: "
1004 << (state.last_sample_is_app_limited ? "yes" : "no");
1005
1006 return os;
1007}
1008
1009} // namespace quic