blob: 205cf65000e123adcf2f14519113d26a2ae61269 [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 <map>
9#include <memory>
10
11#include "net/third_party/quiche/src/quic/core/congestion_control/rtt_stats.h"
12#include "net/third_party/quiche/src/quic/core/quic_packets.h"
13#include "net/third_party/quiche/src/quic/core/quic_utils.h"
14#include "net/third_party/quiche/src/quic/platform/api/quic_logging.h"
15#include "net/third_party/quiche/src/quic/platform/api/quic_ptr_util.h"
16#include "net/third_party/quiche/src/quic/platform/api/quic_test.h"
17#include "net/third_party/quiche/src/quic/test_tools/mock_clock.h"
18#include "net/third_party/quiche/src/quic/test_tools/quic_config_peer.h"
19#include "net/third_party/quiche/src/quic/test_tools/quic_connection_peer.h"
20#include "net/third_party/quiche/src/quic/test_tools/quic_sent_packet_manager_peer.h"
21#include "net/third_party/quiche/src/quic/test_tools/quic_test_utils.h"
22#include "net/third_party/quiche/src/quic/test_tools/simulator/quic_endpoint.h"
23#include "net/third_party/quiche/src/quic/test_tools/simulator/simulator.h"
24#include "net/third_party/quiche/src/quic/test_tools/simulator/switch.h"
25
26namespace quic {
27namespace test {
28
29// Use the initial CWND of 10, as 32 is too much for the test network.
30const uint32_t kInitialCongestionWindowPackets = 10;
31const uint32_t kDefaultWindowTCP =
32 kInitialCongestionWindowPackets * kDefaultTCPMSS;
33
34// Test network parameters. Here, the topology of the network is:
35//
36// BBR sender
37// |
38// | <-- local link (10 Mbps, 2 ms delay)
39// |
40// Network switch
41// * <-- the bottleneck queue in the direction
42// | of the receiver
43// |
44// | <-- test link (4 Mbps, 30 ms delay)
45// |
46// |
47// Receiver
48//
49// The reason the bandwidths chosen are relatively low is the fact that the
50// connection simulator uses QuicTime for its internal clock, and as such has
51// the granularity of 1us, meaning that at bandwidth higher than 20 Mbps the
52// packets can start to land on the same timestamp.
53const QuicBandwidth kTestLinkBandwidth =
54 QuicBandwidth::FromKBitsPerSecond(4000);
55const QuicBandwidth kLocalLinkBandwidth =
56 QuicBandwidth::FromKBitsPerSecond(10000);
57const QuicTime::Delta kTestPropagationDelay =
58 QuicTime::Delta::FromMilliseconds(30);
59const QuicTime::Delta kLocalPropagationDelay =
60 QuicTime::Delta::FromMilliseconds(2);
61const QuicTime::Delta kTestTransferTime =
62 kTestLinkBandwidth.TransferTime(kMaxPacketSize) +
63 kLocalLinkBandwidth.TransferTime(kMaxPacketSize);
64const QuicTime::Delta kTestRtt =
65 (kTestPropagationDelay + kLocalPropagationDelay + kTestTransferTime) * 2;
66const QuicByteCount kTestBdp = kTestRtt * kTestLinkBandwidth;
67
68class BbrSenderTest : public QuicTest {
69 protected:
70 BbrSenderTest()
71 : simulator_(),
72 bbr_sender_(&simulator_,
73 "BBR sender",
74 "Receiver",
75 Perspective::IS_CLIENT,
76 /*connection_id=*/TestConnectionId(42)),
77 competing_sender_(&simulator_,
78 "Competing sender",
79 "Competing receiver",
80 Perspective::IS_CLIENT,
81 /*connection_id=*/TestConnectionId(43)),
82 receiver_(&simulator_,
83 "Receiver",
84 "BBR sender",
85 Perspective::IS_SERVER,
86 /*connection_id=*/TestConnectionId(42)),
87 competing_receiver_(&simulator_,
88 "Competing receiver",
89 "Competing sender",
90 Perspective::IS_SERVER,
91 /*connection_id=*/TestConnectionId(43)),
92 receiver_multiplexer_("Receiver multiplexer",
93 {&receiver_, &competing_receiver_}) {
94 rtt_stats_ = bbr_sender_.connection()->sent_packet_manager().GetRttStats();
95 sender_ = SetupBbrSender(&bbr_sender_);
96
97 clock_ = simulator_.GetClock();
98 simulator_.set_random_generator(&random_);
99
100 uint64_t seed = QuicRandom::GetInstance()->RandUint64();
101 random_.set_seed(seed);
102 QUIC_LOG(INFO) << "BbrSenderTest simulator set up. Seed: " << seed;
103 }
104
105 simulator::Simulator simulator_;
106 simulator::QuicEndpoint bbr_sender_;
107 simulator::QuicEndpoint competing_sender_;
108 simulator::QuicEndpoint receiver_;
109 simulator::QuicEndpoint competing_receiver_;
110 simulator::QuicEndpointMultiplexer receiver_multiplexer_;
111 std::unique_ptr<simulator::Switch> switch_;
112 std::unique_ptr<simulator::SymmetricLink> bbr_sender_link_;
113 std::unique_ptr<simulator::SymmetricLink> competing_sender_link_;
114 std::unique_ptr<simulator::SymmetricLink> receiver_link_;
115
116 SimpleRandom random_;
117
118 // Owned by different components of the connection.
119 const QuicClock* clock_;
120 const RttStats* rtt_stats_;
121 BbrSender* sender_;
122
123 // Enables BBR on |endpoint| and returns the associated BBR congestion
124 // controller.
125 BbrSender* SetupBbrSender(simulator::QuicEndpoint* endpoint) {
126 const RttStats* rtt_stats =
127 endpoint->connection()->sent_packet_manager().GetRttStats();
128 // Ownership of the sender will be overtaken by the endpoint.
129 BbrSender* sender = new BbrSender(
130 rtt_stats,
131 QuicSentPacketManagerPeer::GetUnackedPacketMap(
132 QuicConnectionPeer::GetSentPacketManager(endpoint->connection())),
133 kInitialCongestionWindowPackets, kDefaultMaxCongestionWindowPackets,
134 &random_);
135 QuicConnectionPeer::SetSendAlgorithm(endpoint->connection(), sender);
136 endpoint->RecordTrace();
137 return sender;
138 }
139
140 // Creates a default setup, which is a network with a bottleneck between the
141 // receiver and the switch. The switch has the buffers four times larger than
142 // the bottleneck BDP, which should guarantee a lack of losses.
143 void CreateDefaultSetup() {
144 switch_ = QuicMakeUnique<simulator::Switch>(&simulator_, "Switch", 8,
145 2 * kTestBdp);
146 bbr_sender_link_ = QuicMakeUnique<simulator::SymmetricLink>(
147 &bbr_sender_, switch_->port(1), kLocalLinkBandwidth,
148 kLocalPropagationDelay);
149 receiver_link_ = QuicMakeUnique<simulator::SymmetricLink>(
150 &receiver_, switch_->port(2), kTestLinkBandwidth,
151 kTestPropagationDelay);
152 }
153
154 // Same as the default setup, except the buffer now is half of the BDP.
155 void CreateSmallBufferSetup() {
156 switch_ = QuicMakeUnique<simulator::Switch>(&simulator_, "Switch", 8,
157 0.5 * kTestBdp);
158 bbr_sender_link_ = QuicMakeUnique<simulator::SymmetricLink>(
159 &bbr_sender_, switch_->port(1), kLocalLinkBandwidth,
160 kTestPropagationDelay);
161 receiver_link_ = QuicMakeUnique<simulator::SymmetricLink>(
162 &receiver_, switch_->port(2), kTestLinkBandwidth,
163 kTestPropagationDelay);
164 }
165
166 // Creates the variation of the default setup in which there is another sender
167 // that competes for the same bottleneck link.
168 void CreateCompetitionSetup() {
169 switch_ = QuicMakeUnique<simulator::Switch>(&simulator_, "Switch", 8,
170 2 * kTestBdp);
171
172 // Add a small offset to the competing link in order to avoid
173 // synchronization effects.
174 const QuicTime::Delta small_offset = QuicTime::Delta::FromMicroseconds(3);
175 bbr_sender_link_ = QuicMakeUnique<simulator::SymmetricLink>(
176 &bbr_sender_, switch_->port(1), kLocalLinkBandwidth,
177 kLocalPropagationDelay);
178 competing_sender_link_ = QuicMakeUnique<simulator::SymmetricLink>(
179 &competing_sender_, switch_->port(3), kLocalLinkBandwidth,
180 kLocalPropagationDelay + small_offset);
181 receiver_link_ = QuicMakeUnique<simulator::SymmetricLink>(
182 &receiver_multiplexer_, switch_->port(2), kTestLinkBandwidth,
183 kTestPropagationDelay);
184 }
185
186 // Creates a BBR vs BBR competition setup.
187 void CreateBbrVsBbrSetup() {
188 SetupBbrSender(&competing_sender_);
189 CreateCompetitionSetup();
190 }
191
192 void EnableAggregation(QuicByteCount aggregation_bytes,
193 QuicTime::Delta aggregation_timeout) {
194 // Enable aggregation on the path from the receiver to the sender.
195 switch_->port_queue(1)->EnableAggregation(aggregation_bytes,
196 aggregation_timeout);
197 }
198
199 void DoSimpleTransfer(QuicByteCount transfer_size, QuicTime::Delta deadline) {
200 bbr_sender_.AddBytesToTransfer(transfer_size);
201 // TODO(vasilvv): consider rewriting this to run until the receiver actually
202 // receives the intended amount of bytes.
203 bool simulator_result = simulator_.RunUntilOrTimeout(
204 [this]() { return bbr_sender_.bytes_to_transfer() == 0; }, deadline);
205 EXPECT_TRUE(simulator_result)
206 << "Simple transfer failed. Bytes remaining: "
207 << bbr_sender_.bytes_to_transfer();
208 QUIC_LOG(INFO) << "Simple transfer state: " << sender_->ExportDebugState();
209 }
210
211 // Drive the simulator by sending enough data to enter PROBE_BW.
212 void DriveOutOfStartup() {
213 ASSERT_FALSE(sender_->ExportDebugState().is_at_full_bandwidth);
214 DoSimpleTransfer(1024 * 1024, QuicTime::Delta::FromSeconds(15));
215 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
216 ExpectApproxEq(kTestLinkBandwidth,
217 sender_->ExportDebugState().max_bandwidth, 0.02f);
218 }
219
220 // Send |bytes|-sized bursts of data |number_of_bursts| times, waiting for
221 // |wait_time| between each burst.
222 void SendBursts(size_t number_of_bursts,
223 QuicByteCount bytes,
224 QuicTime::Delta wait_time) {
225 ASSERT_EQ(0u, bbr_sender_.bytes_to_transfer());
226 for (size_t i = 0; i < number_of_bursts; i++) {
227 bbr_sender_.AddBytesToTransfer(bytes);
228
229 // Transfer data and wait for three seconds between each transfer.
230 simulator_.RunFor(wait_time);
231
232 // Ensure the connection did not time out.
233 ASSERT_TRUE(bbr_sender_.connection()->connected());
234 ASSERT_TRUE(receiver_.connection()->connected());
235 }
236
237 simulator_.RunFor(wait_time + kTestRtt);
238 ASSERT_EQ(0u, bbr_sender_.bytes_to_transfer());
239 }
240
241 void SetConnectionOption(QuicTag option) {
242 QuicConfig config;
243 QuicTagVector options;
244 options.push_back(option);
245 QuicConfigPeer::SetReceivedConnectionOptions(&config, options);
246 sender_->SetFromConfig(config, Perspective::IS_SERVER);
247 }
248};
249
250TEST_F(BbrSenderTest, SetInitialCongestionWindow) {
251 EXPECT_NE(3u * kDefaultTCPMSS, sender_->GetCongestionWindow());
252 sender_->SetInitialCongestionWindowInPackets(3);
253 EXPECT_EQ(3u * kDefaultTCPMSS, sender_->GetCongestionWindow());
254}
255
256// Test a simple long data transfer in the default setup.
257TEST_F(BbrSenderTest, SimpleTransfer) {
258 // Disable Ack Decimation on the receiver, because it can increase srtt.
259 QuicConnectionPeer::SetAckMode(receiver_.connection(), AckMode::TCP_ACKING);
260 CreateDefaultSetup();
261
262 // At startup make sure we are at the default.
263 EXPECT_EQ(kDefaultWindowTCP, sender_->GetCongestionWindow());
264 // At startup make sure we can send.
265 EXPECT_TRUE(sender_->CanSend(0));
266 // And that window is un-affected.
267 EXPECT_EQ(kDefaultWindowTCP, sender_->GetCongestionWindow());
268
269 // Verify that Sender is in slow start.
270 EXPECT_TRUE(sender_->InSlowStart());
271
272 // Verify that pacing rate is based on the initial RTT.
273 QuicBandwidth expected_pacing_rate = QuicBandwidth::FromBytesAndTimeDelta(
274 2.885 * kDefaultWindowTCP, rtt_stats_->initial_rtt());
275 ExpectApproxEq(expected_pacing_rate.ToBitsPerSecond(),
276 sender_->PacingRate(0).ToBitsPerSecond(), 0.01f);
277
278 ASSERT_GE(kTestBdp, kDefaultWindowTCP + kDefaultTCPMSS);
279
280 DoSimpleTransfer(12 * 1024 * 1024, QuicTime::Delta::FromSeconds(30));
281 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
282 EXPECT_EQ(0u, bbr_sender_.connection()->GetStats().packets_lost);
283 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
284
285 // The margin here is quite high, since there exists a possibility that the
286 // connection just exited high gain cycle.
287 ExpectApproxEq(kTestRtt, rtt_stats_->smoothed_rtt(), 0.2f);
288}
289
290// Test a simple transfer in a situation when the buffer is less than BDP.
291TEST_F(BbrSenderTest, SimpleTransferSmallBuffer) {
292 CreateSmallBufferSetup();
293
294 DoSimpleTransfer(12 * 1024 * 1024, QuicTime::Delta::FromSeconds(30));
295 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
296 ExpectApproxEq(kTestLinkBandwidth, sender_->ExportDebugState().max_bandwidth,
297 0.01f);
298 EXPECT_GE(bbr_sender_.connection()->GetStats().packets_lost, 0u);
299 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
300}
301
302TEST_F(BbrSenderTest, SimpleTransferEarlyPacketLoss) {
303 SetQuicReloadableFlag(quic_bbr_no_bytes_acked_in_startup_recovery, true);
304 // Enable rate based startup so the recovery window doesn't hide the true
305 // congestion_window_ in GetCongestionWindow().
306 SetConnectionOption(kBBS1);
307 // Disable Ack Decimation on the receiver, because it can increase srtt.
308 QuicConnectionPeer::SetAckMode(receiver_.connection(), AckMode::TCP_ACKING);
309 CreateDefaultSetup();
310
311 // At startup make sure we are at the default.
312 EXPECT_EQ(kDefaultWindowTCP, sender_->GetCongestionWindow());
313 // Verify that Sender is in slow start.
314 EXPECT_TRUE(sender_->InSlowStart());
315 // At startup make sure we can send.
316 EXPECT_TRUE(sender_->CanSend(0));
317 // And that window is un-affected.
318 EXPECT_EQ(kDefaultWindowTCP, sender_->GetCongestionWindow());
319
320 // Transfer 12MB.
321 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
322 // Drop the first packet.
323 receiver_.DropNextIncomingPacket();
324 bool simulator_result = simulator_.RunUntilOrTimeout(
325 [this]() {
326 if (sender_->InRecovery()) {
327 // Two packets are acked before the first is declared lost.
328 EXPECT_LE(sender_->GetCongestionWindow(),
329 (kDefaultWindowTCP + 2 * kDefaultTCPMSS));
330 }
331 return bbr_sender_.bytes_to_transfer() == 0 || !sender_->InSlowStart();
332 },
333 QuicTime::Delta::FromSeconds(30));
334 EXPECT_TRUE(simulator_result) << "Simple transfer failed. Bytes remaining: "
335 << bbr_sender_.bytes_to_transfer();
336 EXPECT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
337 EXPECT_EQ(1u, bbr_sender_.connection()->GetStats().packets_lost);
338 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
339}
340
341// Test a simple long data transfer with 2 rtts of aggregation.
342TEST_F(BbrSenderTest, SimpleTransfer2RTTAggregationBytes) {
343 CreateDefaultSetup();
344 // 2 RTTs of aggregation, with a max of 10kb.
345 EnableAggregation(10 * 1024, 2 * kTestRtt);
346
347 // Transfer 12MB.
348 DoSimpleTransfer(12 * 1024 * 1024, QuicTime::Delta::FromSeconds(35));
349 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
350 // It's possible to read a bandwidth as much as 50% too high with aggregation.
351 EXPECT_LE(kTestLinkBandwidth * 0.99f,
352 sender_->ExportDebugState().max_bandwidth);
353 // TODO(ianswett): Tighten this bound once we understand why BBR is
354 // overestimating bandwidth with aggregation. b/36022633
355 EXPECT_GE(kTestLinkBandwidth * 1.5f,
356 sender_->ExportDebugState().max_bandwidth);
357 // TODO(ianswett): Expect 0 packets are lost once BBR no longer measures
358 // bandwidth higher than the link rate.
359 // The margin here is high, because the aggregation greatly increases
360 // smoothed rtt.
361 EXPECT_GE(kTestRtt * 4, rtt_stats_->smoothed_rtt());
362 ExpectApproxEq(kTestRtt, rtt_stats_->min_rtt(), 0.2f);
363}
364
365// Test a simple long data transfer with 2 rtts of aggregation.
366TEST_F(BbrSenderTest, SimpleTransferAckDecimation) {
367 // Decrease the CWND gain so extra CWND is required with stretch acks.
368 FLAGS_quic_bbr_cwnd_gain = 1.0;
369 sender_ = new BbrSender(
370 rtt_stats_,
371 QuicSentPacketManagerPeer::GetUnackedPacketMap(
372 QuicConnectionPeer::GetSentPacketManager(bbr_sender_.connection())),
373 kInitialCongestionWindowPackets, kDefaultMaxCongestionWindowPackets,
374 &random_);
375 QuicConnectionPeer::SetSendAlgorithm(bbr_sender_.connection(), sender_);
376 // Enable Ack Decimation on the receiver.
377 QuicConnectionPeer::SetAckMode(receiver_.connection(),
378 AckMode::ACK_DECIMATION);
379 CreateDefaultSetup();
380
381 // Transfer 12MB.
382 DoSimpleTransfer(12 * 1024 * 1024, QuicTime::Delta::FromSeconds(35));
383 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
384 // It's possible to read a bandwidth as much as 50% too high with aggregation.
385 EXPECT_LE(kTestLinkBandwidth * 0.99f,
386 sender_->ExportDebugState().max_bandwidth);
387 // TODO(ianswett): Tighten this bound once we understand why BBR is
388 // overestimating bandwidth with aggregation. b/36022633
389 EXPECT_GE(kTestLinkBandwidth * 1.5f,
390 sender_->ExportDebugState().max_bandwidth);
391 // TODO(ianswett): Expect 0 packets are lost once BBR no longer measures
392 // bandwidth higher than the link rate.
393 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
394 // The margin here is high, because the aggregation greatly increases
395 // smoothed rtt.
396 EXPECT_GE(kTestRtt * 2, rtt_stats_->smoothed_rtt());
397 ExpectApproxEq(kTestRtt, rtt_stats_->min_rtt(), 0.1f);
398}
399
400// Test a simple long data transfer with 2 rtts of aggregation.
401TEST_F(BbrSenderTest, SimpleTransfer2RTTAggregationBytes20RTTWindow) {
402 // Disable Ack Decimation on the receiver, because it can increase srtt.
403 QuicConnectionPeer::SetAckMode(receiver_.connection(), AckMode::TCP_ACKING);
404 CreateDefaultSetup();
405 SetConnectionOption(kBBR4);
406 // 2 RTTs of aggregation, with a max of 10kb.
407 EnableAggregation(10 * 1024, 2 * kTestRtt);
408
409 // Transfer 12MB.
410 DoSimpleTransfer(12 * 1024 * 1024, QuicTime::Delta::FromSeconds(35));
411 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
412 // It's possible to read a bandwidth as much as 50% too high with aggregation.
413 EXPECT_LE(kTestLinkBandwidth * 0.99f,
414 sender_->ExportDebugState().max_bandwidth);
415 // TODO(ianswett): Tighten this bound once we understand why BBR is
416 // overestimating bandwidth with aggregation. b/36022633
417 EXPECT_GE(kTestLinkBandwidth * 1.5f,
418 sender_->ExportDebugState().max_bandwidth);
419 // TODO(ianswett): Expect 0 packets are lost once BBR no longer measures
420 // bandwidth higher than the link rate.
421 // The margin here is high, because the aggregation greatly increases
422 // smoothed rtt.
423 EXPECT_GE(kTestRtt * 4, rtt_stats_->smoothed_rtt());
424 ExpectApproxEq(kTestRtt, rtt_stats_->min_rtt(), 0.12f);
425}
426
427// Test a simple long data transfer with 2 rtts of aggregation.
428TEST_F(BbrSenderTest, SimpleTransfer2RTTAggregationBytes40RTTWindow) {
429 // Disable Ack Decimation on the receiver, because it can increase srtt.
430 QuicConnectionPeer::SetAckMode(receiver_.connection(), AckMode::TCP_ACKING);
431 CreateDefaultSetup();
432 SetConnectionOption(kBBR5);
433 // 2 RTTs of aggregation, with a max of 10kb.
434 EnableAggregation(10 * 1024, 2 * kTestRtt);
435
436 // Transfer 12MB.
437 DoSimpleTransfer(12 * 1024 * 1024, QuicTime::Delta::FromSeconds(35));
438 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
439 // It's possible to read a bandwidth as much as 50% too high with aggregation.
440 EXPECT_LE(kTestLinkBandwidth * 0.99f,
441 sender_->ExportDebugState().max_bandwidth);
442 // TODO(ianswett): Tighten this bound once we understand why BBR is
443 // overestimating bandwidth with aggregation. b/36022633
444 EXPECT_GE(kTestLinkBandwidth * 1.5f,
445 sender_->ExportDebugState().max_bandwidth);
446 // TODO(ianswett): Expect 0 packets are lost once BBR no longer measures
447 // bandwidth higher than the link rate.
448 // The margin here is high, because the aggregation greatly increases
449 // smoothed rtt.
450 EXPECT_GE(kTestRtt * 4, rtt_stats_->smoothed_rtt());
451 ExpectApproxEq(kTestRtt, rtt_stats_->min_rtt(), 0.12f);
452}
453
454// Test the number of losses incurred by the startup phase in a situation when
455// the buffer is less than BDP.
456TEST_F(BbrSenderTest, PacketLossOnSmallBufferStartup) {
457 CreateSmallBufferSetup();
458
459 DriveOutOfStartup();
460 float loss_rate =
461 static_cast<float>(bbr_sender_.connection()->GetStats().packets_lost) /
462 bbr_sender_.connection()->GetStats().packets_sent;
463 EXPECT_LE(loss_rate, 0.31);
464}
465
466// Ensures the code transitions loss recovery states correctly (NOT_IN_RECOVERY
467// -> CONSERVATION -> GROWTH -> NOT_IN_RECOVERY).
468TEST_F(BbrSenderTest, RecoveryStates) {
469 // Set seed to the position where the gain cycling causes the sender go
470 // into conservation upon entering PROBE_BW.
471 //
472 // TODO(vasilvv): there should be a better way to test this.
473 random_.set_seed(UINT64_C(14719894707049085006));
474
475 const QuicTime::Delta timeout = QuicTime::Delta::FromSeconds(10);
476 bool simulator_result;
477 CreateSmallBufferSetup();
478
479 bbr_sender_.AddBytesToTransfer(100 * 1024 * 1024);
480 ASSERT_EQ(BbrSender::NOT_IN_RECOVERY,
481 sender_->ExportDebugState().recovery_state);
482
483 simulator_result = simulator_.RunUntilOrTimeout(
484 [this]() {
485 return sender_->ExportDebugState().recovery_state !=
486 BbrSender::NOT_IN_RECOVERY;
487 },
488 timeout);
489 ASSERT_TRUE(simulator_result);
490 ASSERT_EQ(BbrSender::CONSERVATION,
491 sender_->ExportDebugState().recovery_state);
492
493 simulator_result = simulator_.RunUntilOrTimeout(
494 [this]() {
495 return sender_->ExportDebugState().recovery_state !=
496 BbrSender::CONSERVATION;
497 },
498 timeout);
499 ASSERT_TRUE(simulator_result);
500 ASSERT_EQ(BbrSender::GROWTH, sender_->ExportDebugState().recovery_state);
501
502 simulator_result = simulator_.RunUntilOrTimeout(
503 [this]() {
504 return sender_->ExportDebugState().recovery_state != BbrSender::GROWTH;
505 },
506 timeout);
507
508 ASSERT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
509 ASSERT_EQ(BbrSender::NOT_IN_RECOVERY,
510 sender_->ExportDebugState().recovery_state);
511 ASSERT_TRUE(simulator_result);
512}
513
514// Verify the behavior of the algorithm in the case when the connection sends
515// small bursts of data after sending continuously for a while.
516TEST_F(BbrSenderTest, ApplicationLimitedBursts) {
517 CreateDefaultSetup();
518
519 DriveOutOfStartup();
520 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
521
522 SendBursts(20, 512, QuicTime::Delta::FromSeconds(3));
523 EXPECT_TRUE(sender_->ExportDebugState().last_sample_is_app_limited);
524 ExpectApproxEq(kTestLinkBandwidth, sender_->ExportDebugState().max_bandwidth,
525 0.01f);
526}
527
528// Verify the behavior of the algorithm in the case when the connection sends
529// small bursts of data and then starts sending continuously.
530TEST_F(BbrSenderTest, ApplicationLimitedBurstsWithoutPrior) {
531 CreateDefaultSetup();
532
533 SendBursts(40, 512, QuicTime::Delta::FromSeconds(3));
534 EXPECT_TRUE(sender_->ExportDebugState().last_sample_is_app_limited);
535
536 DriveOutOfStartup();
537 ExpectApproxEq(kTestLinkBandwidth, sender_->ExportDebugState().max_bandwidth,
538 0.01f);
539 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
540}
541
542// Verify that the DRAIN phase works correctly.
543TEST_F(BbrSenderTest, Drain) {
544 // Disable Ack Decimation on the receiver, because it can increase srtt.
545 QuicConnectionPeer::SetAckMode(receiver_.connection(), AckMode::TCP_ACKING);
546 CreateDefaultSetup();
547 const QuicTime::Delta timeout = QuicTime::Delta::FromSeconds(10);
548 // Get the queue at the bottleneck, which is the outgoing queue at the port to
549 // which the receiver is connected.
550 const simulator::Queue* queue = switch_->port_queue(2);
551 bool simulator_result;
552
553 // We have no intention of ever finishing this transfer.
554 bbr_sender_.AddBytesToTransfer(100 * 1024 * 1024);
555
556 // Run the startup, and verify that it fills up the queue.
557 ASSERT_EQ(BbrSender::STARTUP, sender_->ExportDebugState().mode);
558 simulator_result = simulator_.RunUntilOrTimeout(
559 [this]() {
560 return sender_->ExportDebugState().mode != BbrSender::STARTUP;
561 },
562 timeout);
563 ASSERT_TRUE(simulator_result);
564 ASSERT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
565 ExpectApproxEq(sender_->BandwidthEstimate() * (1 / 2.885f),
566 sender_->PacingRate(0), 0.01f);
567 // BBR uses CWND gain of 2.88 during STARTUP, hence it will fill the buffer
568 // with approximately 1.88 BDPs. Here, we use 1.5 to give some margin for
569 // error.
570 EXPECT_GE(queue->bytes_queued(), 1.5 * kTestBdp);
571
572 // Observe increased RTT due to bufferbloat.
573 const QuicTime::Delta queueing_delay =
574 kTestLinkBandwidth.TransferTime(queue->bytes_queued());
575 ExpectApproxEq(kTestRtt + queueing_delay, rtt_stats_->latest_rtt(), 0.1f);
576
577 // Transition to the drain phase and verify that it makes the queue
578 // have at most a BDP worth of packets.
579 simulator_result = simulator_.RunUntilOrTimeout(
580 [this]() { return sender_->ExportDebugState().mode != BbrSender::DRAIN; },
581 timeout);
582 ASSERT_TRUE(simulator_result);
583 ASSERT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
584 EXPECT_LE(queue->bytes_queued(), kTestBdp);
585
586 // Wait for a few round trips and ensure we're in appropriate phase of gain
587 // cycling before taking an RTT measurement.
588 const QuicRoundTripCount start_round_trip =
589 sender_->ExportDebugState().round_trip_count;
590 simulator_result = simulator_.RunUntilOrTimeout(
591 [this, start_round_trip]() {
592 QuicRoundTripCount rounds_passed =
593 sender_->ExportDebugState().round_trip_count - start_round_trip;
594 return rounds_passed >= 4 &&
595 sender_->ExportDebugState().gain_cycle_index == 7;
596 },
597 timeout);
598 ASSERT_TRUE(simulator_result);
599
600 // Observe the bufferbloat go away.
601 ExpectApproxEq(kTestRtt, rtt_stats_->smoothed_rtt(), 0.1f);
602}
603
604// Verify that the DRAIN phase works correctly.
605TEST_F(BbrSenderTest, ShallowDrain) {
606 SetQuicReloadableFlag(quic_bbr_slower_startup3, true);
607 // Disable Ack Decimation on the receiver, because it can increase srtt.
608 QuicConnectionPeer::SetAckMode(receiver_.connection(), AckMode::TCP_ACKING);
609
610 CreateDefaultSetup();
611 // BBQ4 increases the pacing gain in DRAIN to 0.75
612 SetConnectionOption(kBBQ4);
613 const QuicTime::Delta timeout = QuicTime::Delta::FromSeconds(10);
614 // Get the queue at the bottleneck, which is the outgoing queue at the port to
615 // which the receiver is connected.
616 const simulator::Queue* queue = switch_->port_queue(2);
617 bool simulator_result;
618
619 // We have no intention of ever finishing this transfer.
620 bbr_sender_.AddBytesToTransfer(100 * 1024 * 1024);
621
622 // Run the startup, and verify that it fills up the queue.
623 ASSERT_EQ(BbrSender::STARTUP, sender_->ExportDebugState().mode);
624 simulator_result = simulator_.RunUntilOrTimeout(
625 [this]() {
626 return sender_->ExportDebugState().mode != BbrSender::STARTUP;
627 },
628 timeout);
629 ASSERT_TRUE(simulator_result);
630 ASSERT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
631 EXPECT_EQ(0.75 * sender_->BandwidthEstimate(), sender_->PacingRate(0));
632 // BBR uses CWND gain of 2.88 during STARTUP, hence it will fill the buffer
633 // with approximately 1.88 BDPs. Here, we use 1.5 to give some margin for
634 // error.
635 EXPECT_GE(queue->bytes_queued(), 1.5 * kTestBdp);
636
637 // Observe increased RTT due to bufferbloat.
638 const QuicTime::Delta queueing_delay =
639 kTestLinkBandwidth.TransferTime(queue->bytes_queued());
640 ExpectApproxEq(kTestRtt + queueing_delay, rtt_stats_->latest_rtt(), 0.1f);
641
642 // Transition to the drain phase and verify that it makes the queue
643 // have at most a BDP worth of packets.
644 simulator_result = simulator_.RunUntilOrTimeout(
645 [this]() { return sender_->ExportDebugState().mode != BbrSender::DRAIN; },
646 timeout);
647 ASSERT_TRUE(simulator_result);
648 ASSERT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
649 EXPECT_LE(queue->bytes_queued(), kTestBdp);
650
651 // Wait for a few round trips and ensure we're in appropriate phase of gain
652 // cycling before taking an RTT measurement.
653 const QuicRoundTripCount start_round_trip =
654 sender_->ExportDebugState().round_trip_count;
655 simulator_result = simulator_.RunUntilOrTimeout(
656 [this, start_round_trip]() {
657 QuicRoundTripCount rounds_passed =
658 sender_->ExportDebugState().round_trip_count - start_round_trip;
659 return rounds_passed >= 4 &&
660 sender_->ExportDebugState().gain_cycle_index == 7;
661 },
662 timeout);
663 ASSERT_TRUE(simulator_result);
664
665 // Observe the bufferbloat go away.
666 ExpectApproxEq(kTestRtt, rtt_stats_->smoothed_rtt(), 0.1f);
667}
668
669// Verify that the connection enters and exits PROBE_RTT correctly.
670TEST_F(BbrSenderTest, ProbeRtt) {
671 CreateDefaultSetup();
672 DriveOutOfStartup();
673
674 // We have no intention of ever finishing this transfer.
675 bbr_sender_.AddBytesToTransfer(100 * 1024 * 1024);
676
677 // Wait until the connection enters PROBE_RTT.
678 const QuicTime::Delta timeout = QuicTime::Delta::FromSeconds(12);
679 bool simulator_result = simulator_.RunUntilOrTimeout(
680 [this]() {
681 return sender_->ExportDebugState().mode == BbrSender::PROBE_RTT;
682 },
683 timeout);
684 ASSERT_TRUE(simulator_result);
685 ASSERT_EQ(BbrSender::PROBE_RTT, sender_->ExportDebugState().mode);
686
687 // Exit PROBE_RTT.
688 const QuicTime probe_rtt_start = clock_->Now();
689 const QuicTime::Delta time_to_exit_probe_rtt =
690 kTestRtt + QuicTime::Delta::FromMilliseconds(200);
691 simulator_.RunFor(1.5 * time_to_exit_probe_rtt);
692 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
693 EXPECT_GE(sender_->ExportDebugState().min_rtt_timestamp, probe_rtt_start);
694}
695
696// Verify that the first sample after PROBE_RTT is not used as the bandwidth,
697// because the round counter doesn't advance during PROBE_RTT.
698TEST_F(BbrSenderTest, AppLimitedRecoveryNoBandwidthDecrease) {
699 SetQuicReloadableFlag(quic_bbr_app_limited_recovery, true);
700 CreateDefaultSetup();
701 DriveOutOfStartup();
702
703 // We have no intention of ever finishing this transfer.
704 bbr_sender_.AddBytesToTransfer(100 * 1024 * 1024);
705
706 // Wait until the connection enters PROBE_RTT.
707 const QuicTime::Delta timeout = QuicTime::Delta::FromSeconds(12);
708 bool simulator_result = simulator_.RunUntilOrTimeout(
709 [this]() {
710 return sender_->ExportDebugState().mode == BbrSender::PROBE_RTT;
711 },
712 timeout);
713 ASSERT_TRUE(simulator_result);
714 ASSERT_EQ(BbrSender::PROBE_RTT, sender_->ExportDebugState().mode);
715
716 const QuicBandwidth beginning_bw = sender_->BandwidthEstimate();
717
718 // Run for most of PROBE_RTT.
719 const QuicTime probe_rtt_start = clock_->Now();
720 const QuicTime::Delta time_to_exit_probe_rtt =
721 kTestRtt + QuicTime::Delta::FromMilliseconds(200);
722 simulator_.RunFor(0.60 * time_to_exit_probe_rtt);
723 EXPECT_EQ(BbrSender::PROBE_RTT, sender_->ExportDebugState().mode);
724 // Lose a packet before exiting PROBE_RTT, which puts us in packet
725 // conservation and then continue there for a while and ensure the bandwidth
726 // estimate doesn't decrease.
727 for (int i = 0; i < 20; ++i) {
728 receiver_.DropNextIncomingPacket();
729 simulator_.RunFor(0.9 * kTestRtt);
730 // Ensure the bandwidth didn't decrease and the samples are app limited.
731 EXPECT_LE(beginning_bw, sender_->BandwidthEstimate());
732 EXPECT_TRUE(sender_->ExportDebugState().last_sample_is_app_limited);
733 }
734 EXPECT_GE(sender_->ExportDebugState().min_rtt_timestamp, probe_rtt_start);
735}
736
737// Verify that the connection enters and exits PROBE_RTT correctly.
738TEST_F(BbrSenderTest, ProbeRttBDPBasedCWNDTarget) {
739 CreateDefaultSetup();
740 SetQuicReloadableFlag(quic_bbr_less_probe_rtt, true);
741 SetConnectionOption(kBBR6);
742 DriveOutOfStartup();
743
744 // We have no intention of ever finishing this transfer.
745 bbr_sender_.AddBytesToTransfer(100 * 1024 * 1024);
746
747 // Wait until the connection enters PROBE_RTT.
748 const QuicTime::Delta timeout = QuicTime::Delta::FromSeconds(12);
749 bool simulator_result = simulator_.RunUntilOrTimeout(
750 [this]() {
751 return sender_->ExportDebugState().mode == BbrSender::PROBE_RTT;
752 },
753 timeout);
754 ASSERT_TRUE(simulator_result);
755 ASSERT_EQ(BbrSender::PROBE_RTT, sender_->ExportDebugState().mode);
756
757 // Exit PROBE_RTT.
758 const QuicTime probe_rtt_start = clock_->Now();
759 const QuicTime::Delta time_to_exit_probe_rtt =
760 kTestRtt + QuicTime::Delta::FromMilliseconds(200);
761 simulator_.RunFor(1.5 * time_to_exit_probe_rtt);
762 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
763 EXPECT_GE(sender_->ExportDebugState().min_rtt_timestamp, probe_rtt_start);
764}
765
766// Verify that the connection enters does not enter PROBE_RTT.
767TEST_F(BbrSenderTest, ProbeRttSkippedAfterAppLimitedAndStableRtt) {
768 CreateDefaultSetup();
769 SetQuicReloadableFlag(quic_bbr_less_probe_rtt, true);
770 SetConnectionOption(kBBR7);
771 DriveOutOfStartup();
772
773 // We have no intention of ever finishing this transfer.
774 bbr_sender_.AddBytesToTransfer(100 * 1024 * 1024);
775
776 // Wait until the connection enters PROBE_RTT.
777 const QuicTime::Delta timeout = QuicTime::Delta::FromSeconds(12);
778 bool simulator_result = simulator_.RunUntilOrTimeout(
779 [this]() {
780 return sender_->ExportDebugState().mode == BbrSender::PROBE_RTT;
781 },
782 timeout);
783 ASSERT_FALSE(simulator_result);
784 ASSERT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
785}
786
787// Verify that the connection enters does not enter PROBE_RTT.
788TEST_F(BbrSenderTest, ProbeRttSkippedAfterAppLimited) {
789 CreateDefaultSetup();
790 SetQuicReloadableFlag(quic_bbr_less_probe_rtt, true);
791 SetConnectionOption(kBBR8);
792 DriveOutOfStartup();
793
794 // We have no intention of ever finishing this transfer.
795 bbr_sender_.AddBytesToTransfer(100 * 1024 * 1024);
796
797 // Wait until the connection enters PROBE_RTT.
798 const QuicTime::Delta timeout = QuicTime::Delta::FromSeconds(12);
799 bool simulator_result = simulator_.RunUntilOrTimeout(
800 [this]() {
801 return sender_->ExportDebugState().mode == BbrSender::PROBE_RTT;
802 },
803 timeout);
804 ASSERT_FALSE(simulator_result);
805 ASSERT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
806}
807
808// Ensure that a connection that is app-limited and is at sufficiently low
809// bandwidth will not exit high gain phase, and similarly ensure that the
810// connection will exit low gain early if the number of bytes in flight is low.
811TEST_F(BbrSenderTest, InFlightAwareGainCycling) {
812 // Disable Ack Decimation on the receiver, because it can increase srtt.
813 QuicConnectionPeer::SetAckMode(receiver_.connection(), AckMode::TCP_ACKING);
814 CreateDefaultSetup();
815 DriveOutOfStartup();
816
817 const QuicTime::Delta timeout = QuicTime::Delta::FromSeconds(5);
818 bool simulator_result;
819
820 // Start a few cycles prior to the high gain one.
821 simulator_result = simulator_.RunUntilOrTimeout(
822 [this]() { return sender_->ExportDebugState().gain_cycle_index == 6; },
823 timeout);
824
825 // Send at 10% of available rate. Run for 3 seconds, checking in the middle
826 // and at the end. The pacing gain should be high throughout.
827 QuicBandwidth target_bandwidth = 0.1f * kTestLinkBandwidth;
828 QuicTime::Delta burst_interval = QuicTime::Delta::FromMilliseconds(300);
829 for (int i = 0; i < 2; i++) {
830 SendBursts(5, target_bandwidth * burst_interval, burst_interval);
831 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
832 EXPECT_EQ(0, sender_->ExportDebugState().gain_cycle_index);
833 ExpectApproxEq(kTestLinkBandwidth,
834 sender_->ExportDebugState().max_bandwidth, 0.01f);
835 }
836
837 // Now that in-flight is almost zero and the pacing gain is still above 1,
838 // send approximately 1.25 BDPs worth of data. This should cause the
839 // PROBE_BW mode to enter low gain cycle, and exit it earlier than one min_rtt
840 // due to running out of data to send.
841 bbr_sender_.AddBytesToTransfer(1.3 * kTestBdp);
842 simulator_result = simulator_.RunUntilOrTimeout(
843 [this]() { return sender_->ExportDebugState().gain_cycle_index == 1; },
844 timeout);
845 ASSERT_TRUE(simulator_result);
846 simulator_.RunFor(0.75 * sender_->ExportDebugState().min_rtt);
847 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
848 EXPECT_EQ(2, sender_->ExportDebugState().gain_cycle_index);
849}
850
851// Ensure that the pacing rate does not drop at startup.
852TEST_F(BbrSenderTest, NoBandwidthDropOnStartup) {
853 CreateDefaultSetup();
854
855 const QuicTime::Delta timeout = QuicTime::Delta::FromSeconds(5);
856 bool simulator_result;
857
858 QuicBandwidth initial_rate = QuicBandwidth::FromBytesAndTimeDelta(
859 kInitialCongestionWindowPackets * kDefaultTCPMSS,
860 rtt_stats_->initial_rtt());
861 EXPECT_GE(sender_->PacingRate(0), initial_rate);
862
863 // Send a packet.
864 bbr_sender_.AddBytesToTransfer(1000);
865 simulator_result = simulator_.RunUntilOrTimeout(
866 [this]() { return receiver_.bytes_received() == 1000; }, timeout);
867 ASSERT_TRUE(simulator_result);
868 EXPECT_GE(sender_->PacingRate(0), initial_rate);
869
870 // Wait for a while.
871 simulator_.RunFor(QuicTime::Delta::FromSeconds(2));
872 EXPECT_GE(sender_->PacingRate(0), initial_rate);
873
874 // Send another packet.
875 bbr_sender_.AddBytesToTransfer(1000);
876 simulator_result = simulator_.RunUntilOrTimeout(
877 [this]() { return receiver_.bytes_received() == 2000; }, timeout);
878 ASSERT_TRUE(simulator_result);
879 EXPECT_GE(sender_->PacingRate(0), initial_rate);
880}
881
882// Test exiting STARTUP earlier due to the 1RTT connection option.
883TEST_F(BbrSenderTest, SimpleTransfer1RTTStartup) {
884 CreateDefaultSetup();
885
886 SetConnectionOption(k1RTT);
887 EXPECT_EQ(1u, sender_->num_startup_rtts());
888
889 // Run until the full bandwidth is reached and check how many rounds it was.
890 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
891 QuicRoundTripCount max_bw_round = 0;
892 QuicBandwidth max_bw(QuicBandwidth::Zero());
893 bool simulator_result = simulator_.RunUntilOrTimeout(
894 [this, &max_bw, &max_bw_round]() {
895 if (max_bw < sender_->ExportDebugState().max_bandwidth) {
896 max_bw = sender_->ExportDebugState().max_bandwidth;
897 max_bw_round = sender_->ExportDebugState().round_trip_count;
898 }
899 return sender_->ExportDebugState().is_at_full_bandwidth;
900 },
901 QuicTime::Delta::FromSeconds(5));
902 ASSERT_TRUE(simulator_result);
903 EXPECT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
904 EXPECT_EQ(1u, sender_->ExportDebugState().round_trip_count - max_bw_round);
905 EXPECT_EQ(1u, sender_->ExportDebugState().rounds_without_bandwidth_gain);
906 EXPECT_EQ(0u, bbr_sender_.connection()->GetStats().packets_lost);
907 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
908}
909
910// Test exiting STARTUP earlier due to the 2RTT connection option.
911TEST_F(BbrSenderTest, SimpleTransfer2RTTStartup) {
912 CreateDefaultSetup();
913
914 SetConnectionOption(k2RTT);
915 EXPECT_EQ(2u, sender_->num_startup_rtts());
916
917 // Run until the full bandwidth is reached and check how many rounds it was.
918 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
919 QuicRoundTripCount max_bw_round = 0;
920 QuicBandwidth max_bw(QuicBandwidth::Zero());
921 bool simulator_result = simulator_.RunUntilOrTimeout(
922 [this, &max_bw, &max_bw_round]() {
923 if (max_bw < sender_->ExportDebugState().max_bandwidth) {
924 max_bw = sender_->ExportDebugState().max_bandwidth;
925 max_bw_round = sender_->ExportDebugState().round_trip_count;
926 }
927 return sender_->ExportDebugState().is_at_full_bandwidth;
928 },
929 QuicTime::Delta::FromSeconds(5));
930 ASSERT_TRUE(simulator_result);
931 EXPECT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
932 EXPECT_EQ(2u, sender_->ExportDebugState().round_trip_count - max_bw_round);
933 EXPECT_EQ(2u, sender_->ExportDebugState().rounds_without_bandwidth_gain);
934 EXPECT_EQ(0u, bbr_sender_.connection()->GetStats().packets_lost);
935 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
936}
937
938// Test exiting STARTUP earlier upon loss due to the LRTT connection option.
939TEST_F(BbrSenderTest, SimpleTransferLRTTStartup) {
940 CreateDefaultSetup();
941
942 SetConnectionOption(kLRTT);
943 EXPECT_EQ(3u, sender_->num_startup_rtts());
944
945 // Run until the full bandwidth is reached and check how many rounds it was.
946 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
947 QuicRoundTripCount max_bw_round = 0;
948 QuicBandwidth max_bw(QuicBandwidth::Zero());
949 bool simulator_result = simulator_.RunUntilOrTimeout(
950 [this, &max_bw, &max_bw_round]() {
951 if (max_bw < sender_->ExportDebugState().max_bandwidth) {
952 max_bw = sender_->ExportDebugState().max_bandwidth;
953 max_bw_round = sender_->ExportDebugState().round_trip_count;
954 }
955 return sender_->ExportDebugState().is_at_full_bandwidth;
956 },
957 QuicTime::Delta::FromSeconds(5));
958 ASSERT_TRUE(simulator_result);
959 EXPECT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
960 EXPECT_EQ(3u, sender_->ExportDebugState().round_trip_count - max_bw_round);
961 EXPECT_EQ(3u, sender_->ExportDebugState().rounds_without_bandwidth_gain);
962 EXPECT_EQ(0u, bbr_sender_.connection()->GetStats().packets_lost);
963 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
964}
965
966// Test exiting STARTUP earlier upon loss due to the LRTT connection option.
967TEST_F(BbrSenderTest, SimpleTransferLRTTStartupSmallBuffer) {
968 CreateSmallBufferSetup();
969
970 SetConnectionOption(kLRTT);
971 EXPECT_EQ(3u, sender_->num_startup_rtts());
972
973 // Run until the full bandwidth is reached and check how many rounds it was.
974 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
975 QuicRoundTripCount max_bw_round = 0;
976 QuicBandwidth max_bw(QuicBandwidth::Zero());
977 bool simulator_result = simulator_.RunUntilOrTimeout(
978 [this, &max_bw, &max_bw_round]() {
979 if (max_bw < sender_->ExportDebugState().max_bandwidth) {
980 max_bw = sender_->ExportDebugState().max_bandwidth;
981 max_bw_round = sender_->ExportDebugState().round_trip_count;
982 }
983 return sender_->ExportDebugState().is_at_full_bandwidth;
984 },
985 QuicTime::Delta::FromSeconds(5));
986 ASSERT_TRUE(simulator_result);
987 EXPECT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
988 EXPECT_GE(2u, sender_->ExportDebugState().round_trip_count - max_bw_round);
989 EXPECT_EQ(1u, sender_->ExportDebugState().rounds_without_bandwidth_gain);
990 EXPECT_NE(0u, bbr_sender_.connection()->GetStats().packets_lost);
991 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
992}
993
994// Test slower pacing after loss in STARTUP due to the BBRS connection option.
995TEST_F(BbrSenderTest, SimpleTransferSlowerStartup) {
996 CreateSmallBufferSetup();
997
998 SetConnectionOption(kBBRS);
999 EXPECT_EQ(3u, sender_->num_startup_rtts());
1000
1001 // Run until the full bandwidth is reached and check how many rounds it was.
1002 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
1003 QuicRoundTripCount max_bw_round = 0;
1004 QuicBandwidth max_bw(QuicBandwidth::Zero());
1005 bool simulator_result = simulator_.RunUntilOrTimeout(
1006 [this, &max_bw, &max_bw_round]() {
1007 if (max_bw < sender_->ExportDebugState().max_bandwidth) {
1008 max_bw = sender_->ExportDebugState().max_bandwidth;
1009 max_bw_round = sender_->ExportDebugState().round_trip_count;
1010 }
1011 // Expect the pacing rate in STARTUP to decrease once packet loss
1012 // is observed, but the CWND does not.
1013 if (bbr_sender_.connection()->GetStats().packets_lost > 0 &&
1014 !sender_->ExportDebugState().is_at_full_bandwidth &&
1015 sender_->has_non_app_limited_sample()) {
1016 EXPECT_EQ(1.5f * max_bw, sender_->PacingRate(0));
1017 }
1018 return sender_->ExportDebugState().is_at_full_bandwidth;
1019 },
1020 QuicTime::Delta::FromSeconds(5));
1021 ASSERT_TRUE(simulator_result);
1022 EXPECT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
1023 EXPECT_GE(3u, sender_->ExportDebugState().round_trip_count - max_bw_round);
1024 EXPECT_EQ(3u, sender_->ExportDebugState().rounds_without_bandwidth_gain);
1025 EXPECT_NE(0u, bbr_sender_.connection()->GetStats().packets_lost);
1026 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
1027}
1028
1029// Ensures no change in congestion window in STARTUP after loss.
1030TEST_F(BbrSenderTest, SimpleTransferNoConservationInStartup) {
1031 CreateSmallBufferSetup();
1032
1033 SetConnectionOption(kBBS1);
1034
1035 // Run until the full bandwidth is reached and check how many rounds it was.
1036 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
1037 bool used_conservation_cwnd = false;
1038 bool simulator_result = simulator_.RunUntilOrTimeout(
1039 [this, &used_conservation_cwnd]() {
1040 if (!sender_->ExportDebugState().is_at_full_bandwidth &&
1041 sender_->GetCongestionWindow() <
1042 sender_->ExportDebugState().congestion_window) {
1043 used_conservation_cwnd = true;
1044 }
1045 return sender_->ExportDebugState().is_at_full_bandwidth;
1046 },
1047 QuicTime::Delta::FromSeconds(5));
1048 ASSERT_TRUE(simulator_result);
1049 EXPECT_FALSE(used_conservation_cwnd);
1050 EXPECT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
1051 EXPECT_EQ(3u, sender_->ExportDebugState().rounds_without_bandwidth_gain);
1052 EXPECT_NE(0u, bbr_sender_.connection()->GetStats().packets_lost);
1053 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
1054}
1055
1056// Ensures no change in congestion window in STARTUP after loss, but that the
1057// rate decreases.
1058TEST_F(BbrSenderTest, SimpleTransferStartupRateReduction) {
1059 SetQuicReloadableFlag(quic_bbr_startup_rate_reduction, true);
1060 CreateSmallBufferSetup();
1061
1062 SetConnectionOption(kBBS4);
1063
1064 // Run until the full bandwidth is reached and check how many rounds it was.
1065 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
1066 bool used_conservation_cwnd = false;
1067 bool simulator_result = simulator_.RunUntilOrTimeout(
1068 [this, &used_conservation_cwnd]() {
1069 if (!sender_->ExportDebugState().is_at_full_bandwidth &&
1070 sender_->GetCongestionWindow() <
1071 sender_->ExportDebugState().congestion_window) {
1072 used_conservation_cwnd = true;
1073 }
1074 // Exit once a loss is hit.
1075 return bbr_sender_.connection()->GetStats().packets_lost > 0 ||
1076 sender_->ExportDebugState().is_at_full_bandwidth;
1077 },
1078 QuicTime::Delta::FromSeconds(5));
1079 ASSERT_TRUE(simulator_result);
1080 EXPECT_TRUE(sender_->InRecovery());
1081 EXPECT_FALSE(used_conservation_cwnd);
1082 EXPECT_EQ(BbrSender::STARTUP, sender_->ExportDebugState().mode);
1083 EXPECT_NE(0u, bbr_sender_.connection()->GetStats().packets_lost);
1084
1085 // Lose each outstanding packet and the pacing rate decreases.
1086 const QuicBandwidth original_pacing_rate = sender_->PacingRate(0);
1087 QuicBandwidth pacing_rate = original_pacing_rate;
1088 const QuicByteCount original_cwnd = sender_->GetCongestionWindow();
1089 LostPacketVector lost_packets;
1090 lost_packets.push_back(LostPacket(QuicPacketNumber(), kMaxPacketSize));
1091 QuicPacketNumber largest_sent =
1092 bbr_sender_.connection()->sent_packet_manager().GetLargestSentPacket();
1093 for (QuicPacketNumber packet_number =
1094 bbr_sender_.connection()->sent_packet_manager().GetLeastUnacked();
1095 packet_number <= largest_sent; ++packet_number) {
1096 lost_packets[0].packet_number = packet_number;
1097 sender_->OnCongestionEvent(false, 0, clock_->Now(), {}, lost_packets);
1098 EXPECT_EQ(original_cwnd, sender_->GetCongestionWindow());
1099 EXPECT_GT(original_pacing_rate, sender_->PacingRate(0));
1100 EXPECT_GE(pacing_rate, sender_->PacingRate(0));
1101 EXPECT_LE(1.25 * sender_->BandwidthEstimate(), sender_->PacingRate(0));
1102 pacing_rate = sender_->PacingRate(0);
1103 }
1104}
1105
1106// Ensures no change in congestion window in STARTUP after loss, but that the
1107// rate decreases twice as fast as BBS4.
1108TEST_F(BbrSenderTest, SimpleTransferDoubleStartupRateReduction) {
1109 SetQuicReloadableFlag(quic_bbr_startup_rate_reduction, true);
1110 CreateSmallBufferSetup();
1111
1112 SetConnectionOption(kBBS5);
1113
1114 // Run until the full bandwidth is reached and check how many rounds it was.
1115 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
1116 bool used_conservation_cwnd = false;
1117 bool simulator_result = simulator_.RunUntilOrTimeout(
1118 [this, &used_conservation_cwnd]() {
1119 if (!sender_->ExportDebugState().is_at_full_bandwidth &&
1120 sender_->GetCongestionWindow() <
1121 sender_->ExportDebugState().congestion_window) {
1122 used_conservation_cwnd = true;
1123 }
1124 // Exit once a loss is hit.
1125 return bbr_sender_.connection()->GetStats().packets_lost > 0 ||
1126 sender_->ExportDebugState().is_at_full_bandwidth;
1127 },
1128 QuicTime::Delta::FromSeconds(5));
1129 ASSERT_TRUE(simulator_result);
1130 EXPECT_TRUE(sender_->InRecovery());
1131 EXPECT_FALSE(used_conservation_cwnd);
1132 EXPECT_EQ(BbrSender::STARTUP, sender_->ExportDebugState().mode);
1133 EXPECT_NE(0u, bbr_sender_.connection()->GetStats().packets_lost);
1134
1135 // Lose each outstanding packet and the pacing rate decreases.
1136 const QuicBandwidth original_pacing_rate = sender_->PacingRate(0);
1137 QuicBandwidth pacing_rate = original_pacing_rate;
1138 const QuicByteCount original_cwnd = sender_->GetCongestionWindow();
1139 LostPacketVector lost_packets;
1140 lost_packets.push_back(LostPacket(QuicPacketNumber(), kMaxPacketSize));
1141 QuicPacketNumber largest_sent =
1142 bbr_sender_.connection()->sent_packet_manager().GetLargestSentPacket();
1143 for (QuicPacketNumber packet_number =
1144 bbr_sender_.connection()->sent_packet_manager().GetLeastUnacked();
1145 packet_number <= largest_sent; ++packet_number) {
1146 lost_packets[0].packet_number = packet_number;
1147 sender_->OnCongestionEvent(false, 0, clock_->Now(), {}, lost_packets);
1148 EXPECT_EQ(original_cwnd, sender_->GetCongestionWindow());
1149 EXPECT_GT(original_pacing_rate, sender_->PacingRate(0));
1150 EXPECT_GE(pacing_rate, sender_->PacingRate(0));
1151 EXPECT_LE(1.25 * sender_->BandwidthEstimate(), sender_->PacingRate(0));
1152 pacing_rate = sender_->PacingRate(0);
1153 }
1154}
1155
1156TEST_F(BbrSenderTest, DerivedPacingGainStartup) {
1157 SetQuicReloadableFlag(quic_bbr_slower_startup3, true);
1158 CreateDefaultSetup();
1159
1160 SetConnectionOption(kBBQ1);
1161 EXPECT_EQ(3u, sender_->num_startup_rtts());
1162 // Verify that Sender is in slow start.
1163 EXPECT_TRUE(sender_->InSlowStart());
1164 // Verify that pacing rate is based on the initial RTT.
1165 QuicBandwidth expected_pacing_rate = QuicBandwidth::FromBytesAndTimeDelta(
1166 2.773 * kDefaultWindowTCP, rtt_stats_->initial_rtt());
1167 ExpectApproxEq(expected_pacing_rate.ToBitsPerSecond(),
1168 sender_->PacingRate(0).ToBitsPerSecond(), 0.01f);
1169
1170 // Run until the full bandwidth is reached and check how many rounds it was.
1171 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
1172 bool simulator_result = simulator_.RunUntilOrTimeout(
1173 [this]() { return sender_->ExportDebugState().is_at_full_bandwidth; },
1174 QuicTime::Delta::FromSeconds(5));
1175 ASSERT_TRUE(simulator_result);
1176 EXPECT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
1177 EXPECT_EQ(3u, sender_->ExportDebugState().rounds_without_bandwidth_gain);
1178 ExpectApproxEq(kTestLinkBandwidth, sender_->ExportDebugState().max_bandwidth,
1179 0.01f);
1180 EXPECT_EQ(0u, bbr_sender_.connection()->GetStats().packets_lost);
1181 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
1182}
1183
1184TEST_F(BbrSenderTest, DerivedCWNDGainStartup) {
1185 SetQuicReloadableFlag(quic_bbr_slower_startup3, true);
1186 CreateDefaultSetup();
1187
1188 SetConnectionOption(kBBQ2);
1189 EXPECT_EQ(3u, sender_->num_startup_rtts());
1190 // Verify that Sender is in slow start.
1191 EXPECT_TRUE(sender_->InSlowStart());
1192 // Verify that pacing rate is based on the initial RTT.
1193 QuicBandwidth expected_pacing_rate = QuicBandwidth::FromBytesAndTimeDelta(
1194 2.885 * kDefaultWindowTCP, rtt_stats_->initial_rtt());
1195 ExpectApproxEq(expected_pacing_rate.ToBitsPerSecond(),
1196 sender_->PacingRate(0).ToBitsPerSecond(), 0.01f);
1197
1198 // Run until the full bandwidth is reached and check how many rounds it was.
1199 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
1200 bool simulator_result = simulator_.RunUntilOrTimeout(
1201 [this]() { return sender_->ExportDebugState().is_at_full_bandwidth; },
1202 QuicTime::Delta::FromSeconds(5));
1203 ASSERT_TRUE(simulator_result);
1204 EXPECT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
1205 EXPECT_EQ(3u, sender_->ExportDebugState().rounds_without_bandwidth_gain);
1206 ExpectApproxEq(kTestLinkBandwidth, sender_->ExportDebugState().max_bandwidth,
1207 0.01f);
1208 EXPECT_EQ(0u, bbr_sender_.connection()->GetStats().packets_lost);
1209 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
1210 // Expect an SRTT less than 2.7 * Min RTT on exit from STARTUP.
1211 EXPECT_GT(kTestRtt * 2.7, rtt_stats_->smoothed_rtt());
1212}
1213
1214TEST_F(BbrSenderTest, AckAggregationInStartup) {
1215 SetQuicReloadableFlag(quic_bbr_slower_startup3, true);
1216 // Disable Ack Decimation on the receiver to avoid loss and make results
1217 // consistent.
1218 QuicConnectionPeer::SetAckMode(receiver_.connection(), AckMode::TCP_ACKING);
1219 CreateDefaultSetup();
1220
1221 SetConnectionOption(kBBQ3);
1222 EXPECT_EQ(3u, sender_->num_startup_rtts());
1223 // Verify that Sender is in slow start.
1224 EXPECT_TRUE(sender_->InSlowStart());
1225 // Verify that pacing rate is based on the initial RTT.
1226 QuicBandwidth expected_pacing_rate = QuicBandwidth::FromBytesAndTimeDelta(
1227 2.885 * kDefaultWindowTCP, rtt_stats_->initial_rtt());
1228 ExpectApproxEq(expected_pacing_rate.ToBitsPerSecond(),
1229 sender_->PacingRate(0).ToBitsPerSecond(), 0.01f);
1230
1231 // Run until the full bandwidth is reached and check how many rounds it was.
1232 bbr_sender_.AddBytesToTransfer(12 * 1024 * 1024);
1233 bool simulator_result = simulator_.RunUntilOrTimeout(
1234 [this]() { return sender_->ExportDebugState().is_at_full_bandwidth; },
1235 QuicTime::Delta::FromSeconds(5));
1236 ASSERT_TRUE(simulator_result);
1237 EXPECT_EQ(BbrSender::DRAIN, sender_->ExportDebugState().mode);
1238 EXPECT_EQ(3u, sender_->ExportDebugState().rounds_without_bandwidth_gain);
1239 ExpectApproxEq(kTestLinkBandwidth, sender_->ExportDebugState().max_bandwidth,
1240 0.01f);
1241 EXPECT_EQ(0u, bbr_sender_.connection()->GetStats().packets_lost);
1242 EXPECT_FALSE(sender_->ExportDebugState().last_sample_is_app_limited);
1243}
1244
1245// Test that two BBR flows started slightly apart from each other terminate.
1246TEST_F(BbrSenderTest, SimpleCompetition) {
1247 const QuicByteCount transfer_size = 10 * 1024 * 1024;
1248 const QuicTime::Delta transfer_time =
1249 kTestLinkBandwidth.TransferTime(transfer_size);
1250 CreateBbrVsBbrSetup();
1251
1252 // Transfer 10% of data in first transfer.
1253 bbr_sender_.AddBytesToTransfer(transfer_size);
1254 bool simulator_result = simulator_.RunUntilOrTimeout(
1255 [this]() { return receiver_.bytes_received() >= 0.1 * transfer_size; },
1256 transfer_time);
1257 ASSERT_TRUE(simulator_result);
1258
1259 // Start the second transfer and wait until both finish.
1260 competing_sender_.AddBytesToTransfer(transfer_size);
1261 simulator_result = simulator_.RunUntilOrTimeout(
1262 [this]() {
1263 return receiver_.bytes_received() == transfer_size &&
1264 competing_receiver_.bytes_received() == transfer_size;
1265 },
1266 3 * transfer_time);
1267 ASSERT_TRUE(simulator_result);
1268}
1269
1270// Test that BBR can resume bandwidth from cached network parameters.
1271TEST_F(BbrSenderTest, ResumeConnectionState) {
1272 CreateDefaultSetup();
1273
1274 bbr_sender_.connection()->AdjustNetworkParameters(kTestLinkBandwidth,
1275 kTestRtt);
1276 EXPECT_EQ(kTestLinkBandwidth, sender_->ExportDebugState().max_bandwidth);
1277 EXPECT_EQ(kTestLinkBandwidth, sender_->BandwidthEstimate());
1278 ExpectApproxEq(kTestRtt, sender_->ExportDebugState().min_rtt, 0.01f);
1279
1280 DriveOutOfStartup();
1281}
1282
1283// Test with a min CWND of 1 instead of 4 packets.
1284TEST_F(BbrSenderTest, ProbeRTTMinCWND1) {
1285 CreateDefaultSetup();
1286 SetConnectionOption(kMIN1);
1287 DriveOutOfStartup();
1288
1289 // We have no intention of ever finishing this transfer.
1290 bbr_sender_.AddBytesToTransfer(100 * 1024 * 1024);
1291
1292 // Wait until the connection enters PROBE_RTT.
1293 const QuicTime::Delta timeout = QuicTime::Delta::FromSeconds(12);
1294 bool simulator_result = simulator_.RunUntilOrTimeout(
1295 [this]() {
1296 return sender_->ExportDebugState().mode == BbrSender::PROBE_RTT;
1297 },
1298 timeout);
1299 ASSERT_TRUE(simulator_result);
1300 ASSERT_EQ(BbrSender::PROBE_RTT, sender_->ExportDebugState().mode);
1301 // The PROBE_RTT CWND should be 1 if the min CWND is 1.
1302 EXPECT_EQ(kDefaultTCPMSS, sender_->GetCongestionWindow());
1303
1304 // Exit PROBE_RTT.
1305 const QuicTime probe_rtt_start = clock_->Now();
1306 const QuicTime::Delta time_to_exit_probe_rtt =
1307 kTestRtt + QuicTime::Delta::FromMilliseconds(200);
1308 simulator_.RunFor(1.5 * time_to_exit_probe_rtt);
1309 EXPECT_EQ(BbrSender::PROBE_BW, sender_->ExportDebugState().mode);
1310 EXPECT_GE(sender_->ExportDebugState().min_rtt_timestamp, probe_rtt_start);
1311}
1312
1313} // namespace test
1314} // namespace quic