| // Copyright 2014 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "net/third_party/quiche/src/quic/core/quic_flow_controller.h" |
| |
| #include <memory> |
| #include <utility> |
| |
| #include "absl/strings/str_cat.h" |
| #include "net/third_party/quiche/src/quic/core/crypto/null_encrypter.h" |
| #include "net/third_party/quiche/src/quic/platform/api/quic_expect_bug.h" |
| #include "net/third_party/quiche/src/quic/platform/api/quic_test.h" |
| #include "net/third_party/quiche/src/quic/test_tools/quic_connection_peer.h" |
| #include "net/third_party/quiche/src/quic/test_tools/quic_flow_controller_peer.h" |
| #include "net/third_party/quiche/src/quic/test_tools/quic_sent_packet_manager_peer.h" |
| #include "net/third_party/quiche/src/quic/test_tools/quic_test_utils.h" |
| |
| using testing::_; |
| using testing::Invoke; |
| |
| namespace quic { |
| namespace test { |
| |
| // Receive window auto-tuning uses RTT in its logic. |
| const int64_t kRtt = 100; |
| |
| class MockFlowController : public QuicFlowControllerInterface { |
| public: |
| MockFlowController() {} |
| MockFlowController(const MockFlowController&) = delete; |
| MockFlowController& operator=(const MockFlowController&) = delete; |
| ~MockFlowController() override {} |
| |
| MOCK_METHOD(void, EnsureWindowAtLeast, (QuicByteCount), (override)); |
| }; |
| |
| class QuicFlowControllerTest : public QuicTest { |
| public: |
| void Initialize() { |
| connection_ = new MockQuicConnection(&helper_, &alarm_factory_, |
| Perspective::IS_CLIENT); |
| connection_->SetEncrypter( |
| ENCRYPTION_FORWARD_SECURE, |
| std::make_unique<NullEncrypter>(connection_->perspective())); |
| session_ = std::make_unique<MockQuicSession>(connection_); |
| flow_controller_ = std::make_unique<QuicFlowController>( |
| session_.get(), stream_id_, /*is_connection_flow_controller*/ false, |
| send_window_, receive_window_, kStreamReceiveWindowLimit, |
| should_auto_tune_receive_window_, &session_flow_controller_); |
| } |
| |
| protected: |
| QuicStreamId stream_id_ = 1234; |
| QuicByteCount send_window_ = kInitialSessionFlowControlWindowForTest; |
| QuicByteCount receive_window_ = kInitialSessionFlowControlWindowForTest; |
| std::unique_ptr<QuicFlowController> flow_controller_; |
| MockQuicConnectionHelper helper_; |
| MockAlarmFactory alarm_factory_; |
| MockQuicConnection* connection_; |
| std::unique_ptr<MockQuicSession> session_; |
| MockFlowController session_flow_controller_; |
| bool should_auto_tune_receive_window_ = false; |
| }; |
| |
| TEST_F(QuicFlowControllerTest, SendingBytes) { |
| Initialize(); |
| |
| EXPECT_FALSE(flow_controller_->IsBlocked()); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(send_window_, flow_controller_->SendWindowSize()); |
| |
| // Send some bytes, but not enough to block. |
| flow_controller_->AddBytesSent(send_window_ / 2); |
| EXPECT_FALSE(flow_controller_->IsBlocked()); |
| EXPECT_EQ(send_window_ / 2, flow_controller_->SendWindowSize()); |
| |
| // Send enough bytes to block. |
| flow_controller_->AddBytesSent(send_window_ / 2); |
| EXPECT_TRUE(flow_controller_->IsBlocked()); |
| EXPECT_EQ(0u, flow_controller_->SendWindowSize()); |
| |
| // BLOCKED frame should get sent. |
| EXPECT_TRUE(flow_controller_->ShouldSendBlocked()); |
| |
| // Update the send window, and verify this has unblocked. |
| EXPECT_TRUE(flow_controller_->UpdateSendWindowOffset(2 * send_window_)); |
| EXPECT_FALSE(flow_controller_->IsBlocked()); |
| EXPECT_EQ(send_window_, flow_controller_->SendWindowSize()); |
| |
| // Updating with a smaller offset doesn't change anything. |
| EXPECT_FALSE(flow_controller_->UpdateSendWindowOffset(send_window_ / 10)); |
| EXPECT_EQ(send_window_, flow_controller_->SendWindowSize()); |
| |
| // Try to send more bytes, violating flow control. |
| EXPECT_CALL(*connection_, |
| CloseConnection(QUIC_FLOW_CONTROL_SENT_TOO_MUCH_DATA, _, _)); |
| EXPECT_QUIC_BUG( |
| flow_controller_->AddBytesSent(send_window_ * 10), |
| absl::StrCat("Trying to send an extra ", send_window_ * 10, " bytes")); |
| EXPECT_TRUE(flow_controller_->IsBlocked()); |
| EXPECT_EQ(0u, flow_controller_->SendWindowSize()); |
| } |
| |
| TEST_F(QuicFlowControllerTest, ReceivingBytes) { |
| Initialize(); |
| |
| EXPECT_FALSE(flow_controller_->IsBlocked()); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| // Receive some bytes, updating highest received offset, but not enough to |
| // fill flow control receive window. |
| EXPECT_TRUE( |
| flow_controller_->UpdateHighestReceivedOffset(1 + receive_window_ / 2)); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ((receive_window_ / 2) - 1, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| // Consume enough bytes to send a WINDOW_UPDATE frame. |
| EXPECT_CALL(*session_, WriteControlFrame(_, _)).Times(1); |
| |
| flow_controller_->AddBytesConsumed(1 + receive_window_ / 2); |
| |
| // Result is that once again we have a fully open receive window. |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| } |
| |
| TEST_F(QuicFlowControllerTest, Move) { |
| Initialize(); |
| |
| flow_controller_->AddBytesSent(send_window_ / 2); |
| EXPECT_FALSE(flow_controller_->IsBlocked()); |
| EXPECT_EQ(send_window_ / 2, flow_controller_->SendWindowSize()); |
| |
| EXPECT_TRUE( |
| flow_controller_->UpdateHighestReceivedOffset(1 + receive_window_ / 2)); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ((receive_window_ / 2) - 1, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| QuicFlowController flow_controller2(std::move(*flow_controller_)); |
| EXPECT_EQ(send_window_ / 2, flow_controller2.SendWindowSize()); |
| EXPECT_FALSE(flow_controller2.FlowControlViolation()); |
| EXPECT_EQ((receive_window_ / 2) - 1, |
| QuicFlowControllerPeer::ReceiveWindowSize(&flow_controller2)); |
| } |
| |
| TEST_F(QuicFlowControllerTest, OnlySendBlockedFrameOncePerOffset) { |
| Initialize(); |
| |
| // Test that we don't send duplicate BLOCKED frames. We should only send one |
| // BLOCKED frame at a given send window offset. |
| EXPECT_FALSE(flow_controller_->IsBlocked()); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(send_window_, flow_controller_->SendWindowSize()); |
| |
| // Send enough bytes to block. |
| flow_controller_->AddBytesSent(send_window_); |
| EXPECT_TRUE(flow_controller_->IsBlocked()); |
| EXPECT_EQ(0u, flow_controller_->SendWindowSize()); |
| |
| // BLOCKED frame should get sent. |
| EXPECT_TRUE(flow_controller_->ShouldSendBlocked()); |
| |
| // BLOCKED frame should not get sent again until our send offset changes. |
| EXPECT_FALSE(flow_controller_->ShouldSendBlocked()); |
| EXPECT_FALSE(flow_controller_->ShouldSendBlocked()); |
| EXPECT_FALSE(flow_controller_->ShouldSendBlocked()); |
| EXPECT_FALSE(flow_controller_->ShouldSendBlocked()); |
| EXPECT_FALSE(flow_controller_->ShouldSendBlocked()); |
| |
| // Update the send window, then send enough bytes to block again. |
| EXPECT_TRUE(flow_controller_->UpdateSendWindowOffset(2 * send_window_)); |
| EXPECT_FALSE(flow_controller_->IsBlocked()); |
| EXPECT_EQ(send_window_, flow_controller_->SendWindowSize()); |
| flow_controller_->AddBytesSent(send_window_); |
| EXPECT_TRUE(flow_controller_->IsBlocked()); |
| EXPECT_EQ(0u, flow_controller_->SendWindowSize()); |
| |
| // BLOCKED frame should get sent as send offset has changed. |
| EXPECT_TRUE(flow_controller_->ShouldSendBlocked()); |
| } |
| |
| TEST_F(QuicFlowControllerTest, ReceivingBytesFastIncreasesFlowWindow) { |
| should_auto_tune_receive_window_ = true; |
| Initialize(); |
| // This test will generate two WINDOW_UPDATE frames. |
| EXPECT_CALL(*session_, WriteControlFrame(_, _)).Times(1); |
| EXPECT_TRUE(flow_controller_->auto_tune_receive_window()); |
| |
| // Make sure clock is inititialized. |
| connection_->AdvanceTime(QuicTime::Delta::FromMilliseconds(1)); |
| |
| QuicSentPacketManager* manager = |
| QuicConnectionPeer::GetSentPacketManager(connection_); |
| |
| RttStats* rtt_stats = const_cast<RttStats*>(manager->GetRttStats()); |
| rtt_stats->UpdateRtt(QuicTime::Delta::FromMilliseconds(kRtt), |
| QuicTime::Delta::Zero(), QuicTime::Zero()); |
| |
| EXPECT_FALSE(flow_controller_->IsBlocked()); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| QuicByteCount threshold = |
| QuicFlowControllerPeer::WindowUpdateThreshold(flow_controller_.get()); |
| |
| QuicStreamOffset receive_offset = threshold + 1; |
| // Receive some bytes, updating highest received offset, but not enough to |
| // fill flow control receive window. |
| EXPECT_TRUE(flow_controller_->UpdateHighestReceivedOffset(receive_offset)); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest - receive_offset, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| EXPECT_CALL( |
| session_flow_controller_, |
| EnsureWindowAtLeast(kInitialSessionFlowControlWindowForTest * 2 * 1.5)); |
| |
| // Consume enough bytes to send a WINDOW_UPDATE frame. |
| flow_controller_->AddBytesConsumed(threshold + 1); |
| // Result is that once again we have a fully open receive window. |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(2 * kInitialSessionFlowControlWindowForTest, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| connection_->AdvanceTime(QuicTime::Delta::FromMilliseconds(2 * kRtt - 1)); |
| receive_offset += threshold + 1; |
| EXPECT_TRUE(flow_controller_->UpdateHighestReceivedOffset(receive_offset)); |
| flow_controller_->AddBytesConsumed(threshold + 1); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| QuicByteCount new_threshold = |
| QuicFlowControllerPeer::WindowUpdateThreshold(flow_controller_.get()); |
| EXPECT_GT(new_threshold, threshold); |
| } |
| |
| TEST_F(QuicFlowControllerTest, ReceivingBytesFastNoAutoTune) { |
| Initialize(); |
| // This test will generate two WINDOW_UPDATE frames. |
| EXPECT_CALL(*session_, WriteControlFrame(_, _)) |
| .Times(2) |
| .WillRepeatedly(Invoke(&ClearControlFrameWithTransmissionType)); |
| EXPECT_FALSE(flow_controller_->auto_tune_receive_window()); |
| |
| // Make sure clock is inititialized. |
| connection_->AdvanceTime(QuicTime::Delta::FromMilliseconds(1)); |
| |
| QuicSentPacketManager* manager = |
| QuicConnectionPeer::GetSentPacketManager(connection_); |
| |
| RttStats* rtt_stats = const_cast<RttStats*>(manager->GetRttStats()); |
| rtt_stats->UpdateRtt(QuicTime::Delta::FromMilliseconds(kRtt), |
| QuicTime::Delta::Zero(), QuicTime::Zero()); |
| |
| EXPECT_FALSE(flow_controller_->IsBlocked()); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| QuicByteCount threshold = |
| QuicFlowControllerPeer::WindowUpdateThreshold(flow_controller_.get()); |
| |
| QuicStreamOffset receive_offset = threshold + 1; |
| // Receive some bytes, updating highest received offset, but not enough to |
| // fill flow control receive window. |
| EXPECT_TRUE(flow_controller_->UpdateHighestReceivedOffset(receive_offset)); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest - receive_offset, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| // Consume enough bytes to send a WINDOW_UPDATE frame. |
| flow_controller_->AddBytesConsumed(threshold + 1); |
| // Result is that once again we have a fully open receive window. |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| // Move time forward, but by less than two RTTs. Then receive and consume |
| // some more, forcing a second WINDOW_UPDATE with an increased max window |
| // size. |
| connection_->AdvanceTime(QuicTime::Delta::FromMilliseconds(2 * kRtt - 1)); |
| receive_offset += threshold + 1; |
| EXPECT_TRUE(flow_controller_->UpdateHighestReceivedOffset(receive_offset)); |
| flow_controller_->AddBytesConsumed(threshold + 1); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| QuicByteCount new_threshold = |
| QuicFlowControllerPeer::WindowUpdateThreshold(flow_controller_.get()); |
| EXPECT_EQ(new_threshold, threshold); |
| } |
| |
| TEST_F(QuicFlowControllerTest, ReceivingBytesNormalStableFlowWindow) { |
| should_auto_tune_receive_window_ = true; |
| Initialize(); |
| // This test will generate two WINDOW_UPDATE frames. |
| EXPECT_CALL(*session_, WriteControlFrame(_, _)).Times(1); |
| EXPECT_TRUE(flow_controller_->auto_tune_receive_window()); |
| |
| // Make sure clock is inititialized. |
| connection_->AdvanceTime(QuicTime::Delta::FromMilliseconds(1)); |
| |
| QuicSentPacketManager* manager = |
| QuicConnectionPeer::GetSentPacketManager(connection_); |
| RttStats* rtt_stats = const_cast<RttStats*>(manager->GetRttStats()); |
| rtt_stats->UpdateRtt(QuicTime::Delta::FromMilliseconds(kRtt), |
| QuicTime::Delta::Zero(), QuicTime::Zero()); |
| |
| EXPECT_FALSE(flow_controller_->IsBlocked()); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| QuicByteCount threshold = |
| QuicFlowControllerPeer::WindowUpdateThreshold(flow_controller_.get()); |
| |
| QuicStreamOffset receive_offset = threshold + 1; |
| // Receive some bytes, updating highest received offset, but not enough to |
| // fill flow control receive window. |
| EXPECT_TRUE(flow_controller_->UpdateHighestReceivedOffset(receive_offset)); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest - receive_offset, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| EXPECT_CALL( |
| session_flow_controller_, |
| EnsureWindowAtLeast(kInitialSessionFlowControlWindowForTest * 2 * 1.5)); |
| flow_controller_->AddBytesConsumed(threshold + 1); |
| |
| // Result is that once again we have a fully open receive window. |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(2 * kInitialSessionFlowControlWindowForTest, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| // Move time forward, but by more than two RTTs. Then receive and consume |
| // some more, forcing a second WINDOW_UPDATE with unchanged max window size. |
| connection_->AdvanceTime(QuicTime::Delta::FromMilliseconds(2 * kRtt + 1)); |
| |
| receive_offset += threshold + 1; |
| EXPECT_TRUE(flow_controller_->UpdateHighestReceivedOffset(receive_offset)); |
| |
| flow_controller_->AddBytesConsumed(threshold + 1); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| |
| QuicByteCount new_threshold = |
| QuicFlowControllerPeer::WindowUpdateThreshold(flow_controller_.get()); |
| EXPECT_EQ(new_threshold, 2 * threshold); |
| } |
| |
| TEST_F(QuicFlowControllerTest, ReceivingBytesNormalNoAutoTune) { |
| Initialize(); |
| // This test will generate two WINDOW_UPDATE frames. |
| EXPECT_CALL(*session_, WriteControlFrame(_, _)) |
| .Times(2) |
| .WillRepeatedly(Invoke(&ClearControlFrameWithTransmissionType)); |
| EXPECT_FALSE(flow_controller_->auto_tune_receive_window()); |
| |
| // Make sure clock is inititialized. |
| connection_->AdvanceTime(QuicTime::Delta::FromMilliseconds(1)); |
| |
| QuicSentPacketManager* manager = |
| QuicConnectionPeer::GetSentPacketManager(connection_); |
| RttStats* rtt_stats = const_cast<RttStats*>(manager->GetRttStats()); |
| rtt_stats->UpdateRtt(QuicTime::Delta::FromMilliseconds(kRtt), |
| QuicTime::Delta::Zero(), QuicTime::Zero()); |
| |
| EXPECT_FALSE(flow_controller_->IsBlocked()); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| QuicByteCount threshold = |
| QuicFlowControllerPeer::WindowUpdateThreshold(flow_controller_.get()); |
| |
| QuicStreamOffset receive_offset = threshold + 1; |
| // Receive some bytes, updating highest received offset, but not enough to |
| // fill flow control receive window. |
| EXPECT_TRUE(flow_controller_->UpdateHighestReceivedOffset(receive_offset)); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest - receive_offset, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| flow_controller_->AddBytesConsumed(threshold + 1); |
| |
| // Result is that once again we have a fully open receive window. |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| EXPECT_EQ(kInitialSessionFlowControlWindowForTest, |
| QuicFlowControllerPeer::ReceiveWindowSize(flow_controller_.get())); |
| |
| // Move time forward, but by more than two RTTs. Then receive and consume |
| // some more, forcing a second WINDOW_UPDATE with unchanged max window size. |
| connection_->AdvanceTime(QuicTime::Delta::FromMilliseconds(2 * kRtt + 1)); |
| |
| receive_offset += threshold + 1; |
| EXPECT_TRUE(flow_controller_->UpdateHighestReceivedOffset(receive_offset)); |
| |
| flow_controller_->AddBytesConsumed(threshold + 1); |
| EXPECT_FALSE(flow_controller_->FlowControlViolation()); |
| |
| QuicByteCount new_threshold = |
| QuicFlowControllerPeer::WindowUpdateThreshold(flow_controller_.get()); |
| |
| EXPECT_EQ(new_threshold, threshold); |
| } |
| |
| } // namespace test |
| } // namespace quic |