mirror of
https://github.com/facebookincubator/mvfst.git
synced 2025-11-12 08:41:01 +03:00
Summary: otherwise we keep encoding packet headers and potentially write into the write buffer without actually generating a packet. It leaves a trail of bad headers inside the buffer Reviewed By: mjoras Differential Revision: D21626733 fbshipit-source-id: 3bdcf6277beccb09b390a590ba2bb0eb8e68e6c1
1298 lines
52 KiB
C++
1298 lines
52 KiB
C++
/*
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
*/
|
|
|
|
#include <quic/api/QuicPacketScheduler.h>
|
|
|
|
#include <folly/portability/GTest.h>
|
|
|
|
#include <quic/api/QuicTransportFunctions.h>
|
|
#include <quic/api/test/Mocks.h>
|
|
#include <quic/client/state/ClientStateMachine.h>
|
|
#include <quic/codec/QuicPacketBuilder.h>
|
|
#include <quic/codec/test/Mocks.h>
|
|
#include <quic/common/test/TestUtils.h>
|
|
#include <quic/fizz/client/handshake/FizzClientQuicHandshakeContext.h>
|
|
#include <quic/server/state/ServerStateMachine.h>
|
|
#include <quic/state/QuicStreamFunctions.h>
|
|
|
|
using namespace quic;
|
|
using namespace testing;
|
|
|
|
namespace {
|
|
|
|
PacketNum addInitialOutstandingPacket(QuicConnectionStateBase& conn) {
|
|
PacketNum nextPacketNum =
|
|
getNextPacketNum(conn, PacketNumberSpace::Handshake);
|
|
std::vector<uint8_t> zeroConnIdData(quic::kDefaultConnectionIdSize, 0);
|
|
ConnectionId srcConnId(zeroConnIdData);
|
|
LongHeader header(
|
|
LongHeader::Types::Initial,
|
|
srcConnId,
|
|
conn.clientConnectionId.value_or(quic::test::getTestConnectionId()),
|
|
nextPacketNum,
|
|
QuicVersion::QUIC_DRAFT);
|
|
RegularQuicWritePacket packet(std::move(header));
|
|
conn.outstandingPackets.emplace_back(packet, Clock::now(), 0, true, 0);
|
|
conn.outstandingHandshakePacketsCount++;
|
|
increaseNextPacketNum(conn, PacketNumberSpace::Handshake);
|
|
return nextPacketNum;
|
|
}
|
|
|
|
PacketNum addHandshakeOutstandingPacket(QuicConnectionStateBase& conn) {
|
|
PacketNum nextPacketNum =
|
|
getNextPacketNum(conn, PacketNumberSpace::Handshake);
|
|
std::vector<uint8_t> zeroConnIdData(quic::kDefaultConnectionIdSize, 0);
|
|
ConnectionId srcConnId(zeroConnIdData);
|
|
LongHeader header(
|
|
LongHeader::Types::Handshake,
|
|
srcConnId,
|
|
conn.clientConnectionId.value_or(quic::test::getTestConnectionId()),
|
|
nextPacketNum,
|
|
QuicVersion::QUIC_DRAFT);
|
|
RegularQuicWritePacket packet(std::move(header));
|
|
conn.outstandingPackets.emplace_back(packet, Clock::now(), 0, true, 0);
|
|
conn.outstandingHandshakePacketsCount++;
|
|
increaseNextPacketNum(conn, PacketNumberSpace::Handshake);
|
|
return nextPacketNum;
|
|
}
|
|
|
|
PacketNum addOutstandingPacket(QuicConnectionStateBase& conn) {
|
|
PacketNum nextPacketNum = getNextPacketNum(conn, PacketNumberSpace::AppData);
|
|
ShortHeader header(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(quic::test::getTestConnectionId()),
|
|
nextPacketNum);
|
|
RegularQuicWritePacket packet(std::move(header));
|
|
conn.outstandingPackets.emplace_back(packet, Clock::now(), 0, false, 0);
|
|
increaseNextPacketNum(conn, PacketNumberSpace::AppData);
|
|
return nextPacketNum;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
namespace quic {
|
|
namespace test {
|
|
|
|
class QuicPacketSchedulerTest : public Test {
|
|
public:
|
|
QuicVersion version{QuicVersion::MVFST};
|
|
};
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CryptoPaddingInitialPacket) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
auto connId = getTestConnectionId();
|
|
LongHeader longHeader(
|
|
LongHeader::Types::Initial,
|
|
getTestConnectionId(1),
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::Initial),
|
|
QuicVersion::MVFST);
|
|
increaseNextPacketNum(conn, PacketNumberSpace::Initial);
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(longHeader),
|
|
conn.ackStates.initialAckState.largestAckedByPeer);
|
|
FrameScheduler cryptoOnlyScheduler =
|
|
std::move(
|
|
FrameScheduler::Builder(
|
|
conn,
|
|
EncryptionLevel::Initial,
|
|
LongHeader::typeToPacketNumberSpace(LongHeader::Types::Initial),
|
|
"CryptoOnlyScheduler")
|
|
.cryptoFrames())
|
|
.build();
|
|
writeDataToQuicStream(
|
|
conn.cryptoState->initialStream, folly::IOBuf::copyBuffer("chlo"));
|
|
auto result = cryptoOnlyScheduler.scheduleFramesForPacket(
|
|
std::move(builder), conn.udpSendPacketLen);
|
|
auto packetLength = result.packet->header->computeChainDataLength() +
|
|
result.packet->body->computeChainDataLength();
|
|
EXPECT_EQ(conn.udpSendPacketLen, packetLength);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, PaddingInitialPureAcks) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
auto connId = getTestConnectionId();
|
|
LongHeader longHeader(
|
|
LongHeader::Types::Initial,
|
|
connId,
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::Initial),
|
|
QuicVersion::MVFST);
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(longHeader),
|
|
conn.ackStates.handshakeAckState.largestAckedByPeer);
|
|
conn.ackStates.initialAckState.largestRecvdPacketTime = Clock::now();
|
|
conn.ackStates.initialAckState.needsToSendAckImmediately = true;
|
|
conn.ackStates.initialAckState.acks.insert(10);
|
|
FrameScheduler acksOnlyScheduler =
|
|
std::move(
|
|
FrameScheduler::Builder(
|
|
conn,
|
|
EncryptionLevel::Initial,
|
|
LongHeader::typeToPacketNumberSpace(LongHeader::Types::Initial),
|
|
"AcksOnlyScheduler")
|
|
.ackFrames())
|
|
.build();
|
|
auto result = acksOnlyScheduler.scheduleFramesForPacket(
|
|
std::move(builder), conn.udpSendPacketLen);
|
|
auto packetLength = result.packet->header->computeChainDataLength() +
|
|
result.packet->body->computeChainDataLength();
|
|
EXPECT_EQ(conn.udpSendPacketLen, packetLength);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, PaddingUpToWrapperSize) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
auto connId = getTestConnectionId();
|
|
size_t cipherOverhead = 30;
|
|
LongHeader longHeader(
|
|
LongHeader::Types::Initial,
|
|
connId,
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::Initial),
|
|
QuicVersion::MVFST);
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(longHeader),
|
|
conn.ackStates.handshakeAckState.largestAckedByPeer);
|
|
conn.ackStates.initialAckState.largestRecvdPacketTime = Clock::now();
|
|
conn.ackStates.initialAckState.needsToSendAckImmediately = true;
|
|
conn.ackStates.initialAckState.acks.insert(10);
|
|
FrameScheduler acksOnlyScheduler =
|
|
std::move(
|
|
FrameScheduler::Builder(
|
|
conn,
|
|
EncryptionLevel::Initial,
|
|
LongHeader::typeToPacketNumberSpace(LongHeader::Types::Initial),
|
|
"AcksOnlyScheduler")
|
|
.ackFrames())
|
|
.build();
|
|
auto result = acksOnlyScheduler.scheduleFramesForPacket(
|
|
std::move(builder), conn.udpSendPacketLen - cipherOverhead);
|
|
auto packetLength = result.packet->header->computeChainDataLength() +
|
|
result.packet->body->computeChainDataLength();
|
|
EXPECT_EQ(conn.udpSendPacketLen - cipherOverhead, packetLength);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CryptoServerInitialPadded) {
|
|
QuicServerConnectionState conn;
|
|
auto connId = getTestConnectionId();
|
|
PacketNum nextPacketNum = getNextPacketNum(conn, PacketNumberSpace::Initial);
|
|
LongHeader longHeader1(
|
|
LongHeader::Types::Initial,
|
|
getTestConnectionId(1),
|
|
connId,
|
|
nextPacketNum,
|
|
QuicVersion::MVFST);
|
|
RegularQuicPacketBuilder builder1(
|
|
conn.udpSendPacketLen,
|
|
std::move(longHeader1),
|
|
conn.ackStates.initialAckState.largestAckedByPeer);
|
|
FrameScheduler scheduler =
|
|
std::move(
|
|
FrameScheduler::Builder(
|
|
conn,
|
|
EncryptionLevel::Initial,
|
|
LongHeader::typeToPacketNumberSpace(LongHeader::Types::Initial),
|
|
"CryptoOnlyScheduler")
|
|
.cryptoFrames())
|
|
.build();
|
|
writeDataToQuicStream(
|
|
conn.cryptoState->initialStream, folly::IOBuf::copyBuffer("shlo"));
|
|
auto result = scheduler.scheduleFramesForPacket(
|
|
std::move(builder1), conn.udpSendPacketLen);
|
|
auto packetLength = result.packet->header->computeChainDataLength() +
|
|
result.packet->body->computeChainDataLength();
|
|
EXPECT_EQ(conn.udpSendPacketLen, packetLength);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, PadTwoInitialPackets) {
|
|
QuicServerConnectionState conn;
|
|
auto connId = getTestConnectionId();
|
|
PacketNum nextPacketNum = getNextPacketNum(conn, PacketNumberSpace::Initial);
|
|
LongHeader longHeader1(
|
|
LongHeader::Types::Initial,
|
|
getTestConnectionId(1),
|
|
connId,
|
|
nextPacketNum,
|
|
QuicVersion::MVFST);
|
|
RegularQuicPacketBuilder builder1(
|
|
conn.udpSendPacketLen,
|
|
std::move(longHeader1),
|
|
conn.ackStates.initialAckState.largestAckedByPeer);
|
|
FrameScheduler scheduler =
|
|
std::move(
|
|
FrameScheduler::Builder(
|
|
conn,
|
|
EncryptionLevel::Initial,
|
|
LongHeader::typeToPacketNumberSpace(LongHeader::Types::Initial),
|
|
"CryptoOnlyScheduler")
|
|
.cryptoFrames())
|
|
.build();
|
|
writeDataToQuicStream(
|
|
conn.cryptoState->initialStream, folly::IOBuf::copyBuffer("shlo"));
|
|
auto result = scheduler.scheduleFramesForPacket(
|
|
std::move(builder1), conn.udpSendPacketLen);
|
|
auto packetLength = result.packet->header->computeChainDataLength() +
|
|
result.packet->body->computeChainDataLength();
|
|
EXPECT_EQ(conn.udpSendPacketLen, packetLength);
|
|
|
|
increaseNextPacketNum(conn, PacketNumberSpace::Initial);
|
|
LongHeader longHeader2(
|
|
LongHeader::Types::Initial,
|
|
getTestConnectionId(1),
|
|
connId,
|
|
nextPacketNum,
|
|
QuicVersion::MVFST);
|
|
RegularQuicPacketBuilder builder2(
|
|
conn.udpSendPacketLen,
|
|
std::move(longHeader2),
|
|
conn.ackStates.initialAckState.largestAckedByPeer);
|
|
writeDataToQuicStream(
|
|
conn.cryptoState->initialStream, folly::IOBuf::copyBuffer("shlo again"));
|
|
auto result2 = scheduler.scheduleFramesForPacket(
|
|
std::move(builder2), conn.udpSendPacketLen);
|
|
packetLength = result2.packet->header->computeChainDataLength() +
|
|
result2.packet->body->computeChainDataLength();
|
|
EXPECT_EQ(conn.udpSendPacketLen, packetLength);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CryptoPaddingRetransmissionClientInitial) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
auto connId = getTestConnectionId();
|
|
LongHeader longHeader(
|
|
LongHeader::Types::Initial,
|
|
getTestConnectionId(1),
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::Initial),
|
|
QuicVersion::MVFST);
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(longHeader),
|
|
conn.ackStates.initialAckState.largestAckedByPeer);
|
|
FrameScheduler scheduler =
|
|
std::move(
|
|
FrameScheduler::Builder(
|
|
conn,
|
|
EncryptionLevel::Initial,
|
|
LongHeader::typeToPacketNumberSpace(LongHeader::Types::Initial),
|
|
"CryptoOnlyScheduler")
|
|
.cryptoFrames())
|
|
.build();
|
|
conn.cryptoState->initialStream.lossBuffer.push_back(
|
|
StreamBuffer{folly::IOBuf::copyBuffer("chlo"), 0, false});
|
|
auto result = scheduler.scheduleFramesForPacket(
|
|
std::move(builder), conn.udpSendPacketLen);
|
|
auto packetLength = result.packet->header->computeChainDataLength() +
|
|
result.packet->body->computeChainDataLength();
|
|
EXPECT_EQ(conn.udpSendPacketLen, packetLength);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CryptoSchedulerOnlySingleLossFits) {
|
|
QuicServerConnectionState conn;
|
|
auto connId = getTestConnectionId();
|
|
LongHeader longHeader(
|
|
LongHeader::Types::Handshake,
|
|
connId,
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::Handshake),
|
|
QuicVersion::MVFST);
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(longHeader),
|
|
conn.ackStates.handshakeAckState.largestAckedByPeer);
|
|
builder.encodePacketHeader();
|
|
PacketBuilderWrapper builderWrapper(builder, 13);
|
|
CryptoStreamScheduler scheduler(
|
|
conn, *getCryptoStream(*conn.cryptoState, EncryptionLevel::Handshake));
|
|
conn.cryptoState->handshakeStream.lossBuffer.push_back(
|
|
StreamBuffer{folly::IOBuf::copyBuffer("shlo"), 0, false});
|
|
conn.cryptoState->handshakeStream.lossBuffer.push_back(StreamBuffer{
|
|
folly::IOBuf::copyBuffer(
|
|
"certificatethatisverylongseriouslythisisextremelylongandcannotfitintoapacket"),
|
|
7,
|
|
false});
|
|
EXPECT_TRUE(scheduler.writeCryptoData(builderWrapper));
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CryptoWritePartialLossBuffer) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
auto connId = getTestConnectionId();
|
|
LongHeader longHeader(
|
|
LongHeader::Types::Initial,
|
|
ConnectionId(std::vector<uint8_t>()),
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::Initial),
|
|
QuicVersion::MVFST);
|
|
RegularQuicPacketBuilder builder(
|
|
25,
|
|
std::move(longHeader),
|
|
conn.ackStates.initialAckState.largestAckedByPeer);
|
|
FrameScheduler cryptoOnlyScheduler =
|
|
std::move(
|
|
FrameScheduler::Builder(
|
|
conn,
|
|
EncryptionLevel::Initial,
|
|
LongHeader::typeToPacketNumberSpace(LongHeader::Types::Initial),
|
|
"CryptoOnlyScheduler")
|
|
.cryptoFrames())
|
|
.build();
|
|
conn.cryptoState->initialStream.lossBuffer.push_back(StreamBuffer{
|
|
folly::IOBuf::copyBuffer("return the special duration value max"),
|
|
0,
|
|
false});
|
|
auto result = cryptoOnlyScheduler.scheduleFramesForPacket(
|
|
std::move(builder), conn.udpSendPacketLen);
|
|
auto packetLength = result.packet->header->computeChainDataLength() +
|
|
result.packet->body->computeChainDataLength();
|
|
EXPECT_LE(packetLength, 25);
|
|
EXPECT_TRUE(result.packet->packet.frames[0].asWriteCryptoFrame() != nullptr);
|
|
EXPECT_FALSE(conn.cryptoState->initialStream.lossBuffer.empty());
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, StreamFrameSchedulerExists) {
|
|
QuicServerConnectionState conn;
|
|
conn.streamManager->setMaxLocalBidirectionalStreams(10);
|
|
auto connId = getTestConnectionId();
|
|
auto stream = conn.streamManager->createNextBidirectionalStream().value();
|
|
|
|
WindowUpdateScheduler scheduler(conn);
|
|
ShortHeader shortHeader(
|
|
ProtectionType::KeyPhaseZero,
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(shortHeader),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
builder.encodePacketHeader();
|
|
auto originalSpace = builder.remainingSpaceInPkt();
|
|
conn.streamManager->queueWindowUpdate(stream->id);
|
|
scheduler.writeWindowUpdates(builder);
|
|
EXPECT_LT(builder.remainingSpaceInPkt(), originalSpace);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, StreamFrameNoSpace) {
|
|
QuicServerConnectionState conn;
|
|
conn.streamManager->setMaxLocalBidirectionalStreams(10);
|
|
auto connId = getTestConnectionId();
|
|
auto stream = conn.streamManager->createNextBidirectionalStream().value();
|
|
|
|
WindowUpdateScheduler scheduler(conn);
|
|
ShortHeader shortHeader(
|
|
ProtectionType::KeyPhaseZero,
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(shortHeader),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
builder.encodePacketHeader();
|
|
PacketBuilderWrapper builderWrapper(builder, 2);
|
|
auto originalSpace = builder.remainingSpaceInPkt();
|
|
conn.streamManager->queueWindowUpdate(stream->id);
|
|
scheduler.writeWindowUpdates(builderWrapper);
|
|
EXPECT_EQ(builder.remainingSpaceInPkt(), originalSpace);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, StreamFrameSchedulerStreamNotExists) {
|
|
QuicServerConnectionState conn;
|
|
auto connId = getTestConnectionId();
|
|
StreamId nonExistentStream = 11;
|
|
|
|
WindowUpdateScheduler scheduler(conn);
|
|
ShortHeader shortHeader(
|
|
ProtectionType::KeyPhaseZero,
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(shortHeader),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
builder.encodePacketHeader();
|
|
auto originalSpace = builder.remainingSpaceInPkt();
|
|
conn.streamManager->queueWindowUpdate(nonExistentStream);
|
|
scheduler.writeWindowUpdates(builder);
|
|
EXPECT_EQ(builder.remainingSpaceInPkt(), originalSpace);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CloningSchedulerTest) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
FrameScheduler noopScheduler("frame");
|
|
ASSERT_FALSE(noopScheduler.hasData());
|
|
CloningScheduler cloningScheduler(noopScheduler, conn, "CopyCat", 0);
|
|
EXPECT_FALSE(cloningScheduler.hasData());
|
|
auto packetNum = addOutstandingPacket(conn);
|
|
// There needs to have retransmittable frame for the rebuilder to work
|
|
conn.outstandingPackets.back().packet.frames.push_back(
|
|
MaxDataFrame(conn.flowControlState.advertisedMaxOffset));
|
|
EXPECT_TRUE(cloningScheduler.hasData());
|
|
|
|
ASSERT_FALSE(noopScheduler.hasData());
|
|
ShortHeader header(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(header),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
auto result = cloningScheduler.scheduleFramesForPacket(
|
|
std::move(builder), kDefaultUDPSendPacketLen);
|
|
EXPECT_TRUE(result.packetEvent.has_value() && result.packet.has_value());
|
|
EXPECT_EQ(packetNum, *result.packetEvent);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, DoNotCloneProcessedClonedPacket) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
FrameScheduler noopScheduler("frame");
|
|
CloningScheduler cloningScheduler(noopScheduler, conn, "CopyCat", 0);
|
|
// Add two outstanding packets, but then mark the second one processed by
|
|
// adding a PacketEvent that's missing from the outstandingPacketEvents set
|
|
PacketNum expected = addOutstandingPacket(conn);
|
|
// There needs to have retransmittable frame for the rebuilder to work
|
|
conn.outstandingPackets.back().packet.frames.push_back(
|
|
MaxDataFrame(conn.flowControlState.advertisedMaxOffset));
|
|
addOutstandingPacket(conn);
|
|
conn.outstandingPackets.back().associatedEvent = 1;
|
|
// There needs to have retransmittable frame for the rebuilder to work
|
|
conn.outstandingPackets.back().packet.frames.push_back(
|
|
MaxDataFrame(conn.flowControlState.advertisedMaxOffset));
|
|
|
|
ShortHeader header(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(header),
|
|
conn.ackStates.initialAckState.largestAckedByPeer);
|
|
auto result = cloningScheduler.scheduleFramesForPacket(
|
|
std::move(builder), kDefaultUDPSendPacketLen);
|
|
EXPECT_TRUE(result.packetEvent.has_value() && result.packet.has_value());
|
|
EXPECT_EQ(expected, *result.packetEvent);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CloneSchedulerHasDataIgnoresNonAppData) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
FrameScheduler noopScheduler("frame");
|
|
CloningScheduler cloningScheduler(noopScheduler, conn, "CopyCat", 0);
|
|
EXPECT_FALSE(cloningScheduler.hasData());
|
|
|
|
addHandshakeOutstandingPacket(conn);
|
|
EXPECT_FALSE(cloningScheduler.hasData());
|
|
|
|
addInitialOutstandingPacket(conn);
|
|
EXPECT_FALSE(cloningScheduler.hasData());
|
|
|
|
addOutstandingPacket(conn);
|
|
EXPECT_TRUE(cloningScheduler.hasData());
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, DoNotCloneHandshake) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
FrameScheduler noopScheduler("frame");
|
|
CloningScheduler cloningScheduler(noopScheduler, conn, "CopyCat", 0);
|
|
// Add two outstanding packets, with second one being handshake
|
|
auto expected = addOutstandingPacket(conn);
|
|
// There needs to have retransmittable frame for the rebuilder to work
|
|
conn.outstandingPackets.back().packet.frames.push_back(
|
|
MaxDataFrame(conn.flowControlState.advertisedMaxOffset));
|
|
addHandshakeOutstandingPacket(conn);
|
|
conn.outstandingPackets.back().packet.frames.push_back(
|
|
MaxDataFrame(conn.flowControlState.advertisedMaxOffset));
|
|
|
|
ShortHeader header(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(header),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
auto result = cloningScheduler.scheduleFramesForPacket(
|
|
std::move(builder), kDefaultUDPSendPacketLen);
|
|
EXPECT_TRUE(result.packetEvent.has_value() && result.packet.has_value());
|
|
EXPECT_EQ(expected, *result.packetEvent);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CloneSchedulerUseNormalSchedulerFirst) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
NiceMock<MockFrameScheduler> mockScheduler;
|
|
CloningScheduler cloningScheduler(mockScheduler, conn, "Mocker", 0);
|
|
ShortHeader header(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
EXPECT_CALL(mockScheduler, hasData()).Times(1).WillOnce(Return(true));
|
|
|
|
EXPECT_CALL(mockScheduler, _scheduleFramesForPacket(_, _))
|
|
.Times(1)
|
|
.WillOnce(Invoke(
|
|
[&, headerCopy = header](PacketBuilderInterface*, uint32_t) mutable {
|
|
RegularQuicWritePacket packet(std::move(headerCopy));
|
|
packet.frames.push_back(MaxDataFrame(2832));
|
|
RegularQuicPacketBuilder::Packet builtPacket(
|
|
std::move(packet),
|
|
folly::IOBuf::copyBuffer("if you are the dealer"),
|
|
folly::IOBuf::copyBuffer("I'm out of the game"));
|
|
return SchedulingResult(folly::none, std::move(builtPacket));
|
|
}));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(header),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
auto result = cloningScheduler.scheduleFramesForPacket(
|
|
std::move(builder), kDefaultUDPSendPacketLen);
|
|
EXPECT_EQ(folly::none, result.packetEvent);
|
|
EXPECT_EQ(result.packet->packet.header.getHeaderForm(), HeaderForm::Short);
|
|
ShortHeader& shortHeader = *result.packet->packet.header.asShort();
|
|
EXPECT_EQ(ProtectionType::KeyPhaseOne, shortHeader.getProtectionType());
|
|
EXPECT_EQ(
|
|
conn.ackStates.appDataAckState.nextPacketNum,
|
|
shortHeader.getPacketSequenceNum());
|
|
EXPECT_EQ(1, result.packet->packet.frames.size());
|
|
MaxDataFrame* maxDataFrame =
|
|
result.packet->packet.frames.front().asMaxDataFrame();
|
|
ASSERT_NE(maxDataFrame, nullptr);
|
|
EXPECT_EQ(2832, maxDataFrame->maximumData);
|
|
EXPECT_TRUE(folly::IOBufEqualTo{}(
|
|
*folly::IOBuf::copyBuffer("if you are the dealer"),
|
|
*result.packet->header));
|
|
EXPECT_TRUE(folly::IOBufEqualTo{}(
|
|
*folly::IOBuf::copyBuffer("I'm out of the game"), *result.packet->body));
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CloneWillGenerateNewWindowUpdate) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.streamManager->setMaxLocalBidirectionalStreams(10);
|
|
auto stream = conn.streamManager->createNextBidirectionalStream().value();
|
|
FrameScheduler noopScheduler("frame");
|
|
CloningScheduler cloningScheduler(noopScheduler, conn, "GiantsShoulder", 0);
|
|
auto expectedPacketEvent = addOutstandingPacket(conn);
|
|
ASSERT_EQ(1, conn.outstandingPackets.size());
|
|
conn.outstandingPackets.back().packet.frames.push_back(MaxDataFrame(1000));
|
|
conn.outstandingPackets.back().packet.frames.push_back(
|
|
MaxStreamDataFrame(stream->id, 1000));
|
|
conn.flowControlState.advertisedMaxOffset = 1000;
|
|
stream->flowControlState.advertisedMaxOffset = 1000;
|
|
|
|
conn.flowControlState.sumCurReadOffset = 300;
|
|
conn.flowControlState.windowSize = 3000;
|
|
stream->currentReadOffset = 200;
|
|
stream->flowControlState.windowSize = 1500;
|
|
|
|
ShortHeader header(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(header),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
auto packetResult = cloningScheduler.scheduleFramesForPacket(
|
|
std::move(builder), conn.udpSendPacketLen);
|
|
EXPECT_EQ(expectedPacketEvent, *packetResult.packetEvent);
|
|
int32_t verifyConnWindowUpdate = 1, verifyStreamWindowUpdate = 1;
|
|
for (const auto& frame : packetResult.packet->packet.frames) {
|
|
switch (frame.type()) {
|
|
case QuicWriteFrame::Type::MaxStreamDataFrame_E: {
|
|
const MaxStreamDataFrame& maxStreamDataFrame =
|
|
*frame.asMaxStreamDataFrame();
|
|
EXPECT_EQ(stream->id, maxStreamDataFrame.streamId);
|
|
verifyStreamWindowUpdate--;
|
|
break;
|
|
}
|
|
case QuicWriteFrame::Type::MaxDataFrame_E: {
|
|
verifyConnWindowUpdate--;
|
|
break;
|
|
}
|
|
case QuicWriteFrame::Type::PaddingFrame_E: {
|
|
break;
|
|
}
|
|
default:
|
|
// should never happen
|
|
EXPECT_TRUE(false);
|
|
}
|
|
}
|
|
EXPECT_EQ(0, verifyStreamWindowUpdate);
|
|
EXPECT_EQ(0, verifyConnWindowUpdate);
|
|
|
|
// Verify the built out packet has refreshed window update values
|
|
EXPECT_GE(packetResult.packet->packet.frames.size(), 2);
|
|
uint32_t streamWindowUpdateCounter = 0;
|
|
uint32_t connWindowUpdateCounter = 0;
|
|
for (auto& frame : packetResult.packet->packet.frames) {
|
|
auto streamFlowControl = frame.asMaxStreamDataFrame();
|
|
if (!streamFlowControl) {
|
|
continue;
|
|
}
|
|
streamWindowUpdateCounter++;
|
|
EXPECT_EQ(1700, streamFlowControl->maximumData);
|
|
}
|
|
for (auto& frame : packetResult.packet->packet.frames) {
|
|
auto connFlowControl = frame.asMaxDataFrame();
|
|
if (!connFlowControl) {
|
|
continue;
|
|
}
|
|
connWindowUpdateCounter++;
|
|
EXPECT_EQ(3300, connFlowControl->maximumData);
|
|
}
|
|
EXPECT_EQ(1, connWindowUpdateCounter);
|
|
EXPECT_EQ(1, streamWindowUpdateCounter);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CloningSchedulerWithInplaceBuilder) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.transportSettings.dataPathType = DataPathType::ContinuousMemory;
|
|
SimpleBufAccessor bufAccessor(2000);
|
|
auto buf = bufAccessor.obtain();
|
|
EXPECT_EQ(buf->length(), 0);
|
|
bufAccessor.release(std::move(buf));
|
|
conn.bufAccessor = &bufAccessor;
|
|
|
|
FrameScheduler noopScheduler("frame");
|
|
ASSERT_FALSE(noopScheduler.hasData());
|
|
CloningScheduler cloningScheduler(noopScheduler, conn, "93MillionMiles", 0);
|
|
auto packetNum = addOutstandingPacket(conn);
|
|
// There needs to have retransmittable frame for the rebuilder to work
|
|
conn.outstandingPackets.back().packet.frames.push_back(
|
|
MaxDataFrame(conn.flowControlState.advertisedMaxOffset));
|
|
EXPECT_TRUE(cloningScheduler.hasData());
|
|
|
|
ASSERT_FALSE(noopScheduler.hasData());
|
|
ShortHeader header(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
InplaceQuicPacketBuilder builder(
|
|
bufAccessor,
|
|
conn.udpSendPacketLen,
|
|
std::move(header),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
auto result = cloningScheduler.scheduleFramesForPacket(
|
|
std::move(builder), kDefaultUDPSendPacketLen);
|
|
EXPECT_TRUE(result.packetEvent.has_value() && result.packet.has_value());
|
|
EXPECT_EQ(packetNum, *result.packetEvent);
|
|
|
|
// Something was written into the buffer:
|
|
EXPECT_TRUE(bufAccessor.ownsBuffer());
|
|
buf = bufAccessor.obtain();
|
|
EXPECT_GT(buf->length(), 10);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CloningSchedulerWithInplaceBuilderFullPacket) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.streamManager->setMaxLocalBidirectionalStreams(10);
|
|
conn.flowControlState.peerAdvertisedMaxOffset = 100000;
|
|
conn.flowControlState.peerAdvertisedInitialMaxStreamOffsetBidiRemote = 100000;
|
|
conn.transportSettings.dataPathType = DataPathType::ContinuousMemory;
|
|
SimpleBufAccessor bufAccessor(2000);
|
|
auto buf = bufAccessor.obtain();
|
|
EXPECT_EQ(buf->length(), 0);
|
|
bufAccessor.release(std::move(buf));
|
|
conn.bufAccessor = &bufAccessor;
|
|
auto stream = *conn.streamManager->createNextBidirectionalStream();
|
|
auto inBuf = buildRandomInputData(conn.udpSendPacketLen * 10);
|
|
writeDataToQuicStream(*stream, inBuf->clone(), false);
|
|
|
|
FrameScheduler scheduler = std::move(FrameScheduler::Builder(
|
|
conn,
|
|
EncryptionLevel::AppData,
|
|
PacketNumberSpace::AppData,
|
|
"streamScheduler")
|
|
.streamFrames())
|
|
.build();
|
|
auto packetNum = getNextPacketNum(conn, PacketNumberSpace::AppData);
|
|
ShortHeader header(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
packetNum);
|
|
InplaceQuicPacketBuilder builder(
|
|
bufAccessor,
|
|
conn.udpSendPacketLen,
|
|
std::move(header),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
ASSERT_TRUE(scheduler.hasData());
|
|
auto result = scheduler.scheduleFramesForPacket(
|
|
std::move(builder), conn.udpSendPacketLen);
|
|
auto bufferLength = result.packet->header->computeChainDataLength() +
|
|
result.packet->body->computeChainDataLength();
|
|
EXPECT_EQ(conn.udpSendPacketLen, bufferLength);
|
|
updateConnection(
|
|
conn, folly::none, result.packet->packet, Clock::now(), bufferLength);
|
|
buf = bufAccessor.obtain();
|
|
ASSERT_EQ(conn.udpSendPacketLen, buf->length());
|
|
buf->clear();
|
|
bufAccessor.release(std::move(buf));
|
|
|
|
FrameScheduler noopScheduler("noopScheduler");
|
|
ASSERT_FALSE(noopScheduler.hasData());
|
|
CloningScheduler cloningScheduler(noopScheduler, conn, "93MillionMiles", 0);
|
|
EXPECT_TRUE(cloningScheduler.hasData());
|
|
ASSERT_FALSE(noopScheduler.hasData());
|
|
// Exact same header, so header encoding should be the same
|
|
ShortHeader dupHeader(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
packetNum);
|
|
InplaceQuicPacketBuilder internalBuilder(
|
|
bufAccessor,
|
|
conn.udpSendPacketLen,
|
|
std::move(dupHeader),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
auto cloneResult = cloningScheduler.scheduleFramesForPacket(
|
|
std::move(internalBuilder), conn.udpSendPacketLen);
|
|
EXPECT_TRUE(
|
|
cloneResult.packetEvent.has_value() && cloneResult.packet.has_value());
|
|
EXPECT_EQ(packetNum, *cloneResult.packetEvent);
|
|
|
|
// Something was written into the buffer:
|
|
EXPECT_TRUE(bufAccessor.ownsBuffer());
|
|
buf = bufAccessor.obtain();
|
|
EXPECT_EQ(buf->length(), conn.udpSendPacketLen);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, CloneLargerThanOriginalPacket) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.udpSendPacketLen = 1000;
|
|
conn.streamManager->setMaxLocalBidirectionalStreams(10);
|
|
conn.flowControlState.peerAdvertisedMaxOffset = 100000;
|
|
conn.flowControlState.peerAdvertisedInitialMaxStreamOffsetBidiRemote = 100000;
|
|
auto stream = conn.streamManager->createNextBidirectionalStream().value();
|
|
auto inputData = buildRandomInputData(conn.udpSendPacketLen * 10);
|
|
writeDataToQuicStream(*stream, inputData->clone(), false);
|
|
FrameScheduler scheduler = std::move(FrameScheduler::Builder(
|
|
conn,
|
|
EncryptionLevel::AppData,
|
|
PacketNumberSpace::AppData,
|
|
"streamScheduler")
|
|
.streamFrames())
|
|
.build();
|
|
auto cipherOverhead = 16;
|
|
PacketNum packetNum = 0;
|
|
ShortHeader header(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
packetNum);
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(header),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
auto packetResult = scheduler.scheduleFramesForPacket(
|
|
std::move(builder), conn.udpSendPacketLen - cipherOverhead);
|
|
auto encodedSize = packetResult.packet->body->computeChainDataLength() +
|
|
packetResult.packet->header->computeChainDataLength() + cipherOverhead;
|
|
EXPECT_EQ(encodedSize, conn.udpSendPacketLen);
|
|
updateConnection(
|
|
conn,
|
|
folly::none,
|
|
packetResult.packet->packet,
|
|
Clock::now(),
|
|
encodedSize);
|
|
|
|
// make packetNum too larger to be encoded into the same size:
|
|
packetNum += 0xFF;
|
|
ShortHeader cloneHeader(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
packetNum);
|
|
RegularQuicPacketBuilder throwawayBuilder(
|
|
conn.udpSendPacketLen,
|
|
std::move(cloneHeader),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
FrameScheduler noopScheduler("noopScheduler");
|
|
CloningScheduler cloningScheduler(
|
|
noopScheduler, conn, "CopyCat", cipherOverhead);
|
|
auto cloneResult = cloningScheduler.scheduleFramesForPacket(
|
|
std::move(throwawayBuilder), kDefaultUDPSendPacketLen);
|
|
EXPECT_FALSE(cloneResult.packet.hasValue());
|
|
EXPECT_FALSE(cloneResult.packetEvent.hasValue());
|
|
}
|
|
|
|
class AckSchedulingTest : public TestWithParam<PacketNumberSpace> {};
|
|
|
|
TEST_F(QuicPacketSchedulerTest, AckStateHasAcksToSchedule) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
EXPECT_FALSE(hasAcksToSchedule(conn.ackStates.initialAckState));
|
|
EXPECT_FALSE(hasAcksToSchedule(conn.ackStates.handshakeAckState));
|
|
EXPECT_FALSE(hasAcksToSchedule(conn.ackStates.appDataAckState));
|
|
|
|
conn.ackStates.initialAckState.acks.insert(0, 100);
|
|
EXPECT_TRUE(hasAcksToSchedule(conn.ackStates.initialAckState));
|
|
|
|
conn.ackStates.handshakeAckState.acks.insert(0, 100);
|
|
conn.ackStates.handshakeAckState.largestAckScheduled = 200;
|
|
EXPECT_FALSE(hasAcksToSchedule(conn.ackStates.handshakeAckState));
|
|
|
|
conn.ackStates.handshakeAckState.largestAckScheduled = folly::none;
|
|
EXPECT_TRUE(hasAcksToSchedule(conn.ackStates.handshakeAckState));
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, AckSchedulerHasAcksToSchedule) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
AckScheduler initialAckScheduler(
|
|
conn, getAckState(conn, PacketNumberSpace::Initial));
|
|
AckScheduler handshakeAckScheduler(
|
|
conn, getAckState(conn, PacketNumberSpace::Handshake));
|
|
AckScheduler appDataAckScheduler(
|
|
conn, getAckState(conn, PacketNumberSpace::AppData));
|
|
EXPECT_FALSE(initialAckScheduler.hasPendingAcks());
|
|
EXPECT_FALSE(handshakeAckScheduler.hasPendingAcks());
|
|
EXPECT_FALSE(appDataAckScheduler.hasPendingAcks());
|
|
|
|
conn.ackStates.initialAckState.acks.insert(0, 100);
|
|
EXPECT_TRUE(initialAckScheduler.hasPendingAcks());
|
|
|
|
conn.ackStates.handshakeAckState.acks.insert(0, 100);
|
|
conn.ackStates.handshakeAckState.largestAckScheduled = 200;
|
|
EXPECT_FALSE(handshakeAckScheduler.hasPendingAcks());
|
|
|
|
conn.ackStates.handshakeAckState.largestAckScheduled = folly::none;
|
|
EXPECT_TRUE(handshakeAckScheduler.hasPendingAcks());
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, LargestAckToSend) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
EXPECT_EQ(folly::none, largestAckToSend(conn.ackStates.initialAckState));
|
|
EXPECT_EQ(folly::none, largestAckToSend(conn.ackStates.handshakeAckState));
|
|
EXPECT_EQ(folly::none, largestAckToSend(conn.ackStates.appDataAckState));
|
|
|
|
conn.ackStates.initialAckState.acks.insert(0, 50);
|
|
conn.ackStates.handshakeAckState.acks.insert(0, 50);
|
|
conn.ackStates.handshakeAckState.acks.insert(75, 150);
|
|
|
|
EXPECT_EQ(50, *largestAckToSend(conn.ackStates.initialAckState));
|
|
EXPECT_EQ(150, *largestAckToSend(conn.ackStates.handshakeAckState));
|
|
EXPECT_EQ(folly::none, largestAckToSend(conn.ackStates.appDataAckState));
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, StreamFrameSchedulerAllFit) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.streamManager->setMaxLocalBidirectionalStreams(10);
|
|
conn.flowControlState.peerAdvertisedMaxOffset = 100000;
|
|
conn.flowControlState.peerAdvertisedInitialMaxStreamOffsetBidiRemote = 100000;
|
|
auto connId = getTestConnectionId();
|
|
StreamFrameScheduler scheduler(conn);
|
|
ShortHeader shortHeader(
|
|
ProtectionType::KeyPhaseZero,
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(shortHeader),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
builder.encodePacketHeader();
|
|
auto stream1 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto stream2 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto stream3 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream1),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream2),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream3),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
scheduler.writeStreams(builder);
|
|
EXPECT_EQ(conn.schedulingState.nextScheduledStream, 0);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, StreamFrameSchedulerRoundRobin) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.streamManager->setMaxLocalBidirectionalStreams(10);
|
|
conn.flowControlState.peerAdvertisedMaxOffset = 100000;
|
|
conn.flowControlState.peerAdvertisedInitialMaxStreamOffsetBidiRemote = 100000;
|
|
auto connId = getTestConnectionId();
|
|
StreamFrameScheduler scheduler(conn);
|
|
ShortHeader shortHeader1(
|
|
ProtectionType::KeyPhaseZero,
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(shortHeader1),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
builder.encodePacketHeader();
|
|
auto stream1 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto stream2 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto stream3 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto largeBuf = folly::IOBuf::createChain(conn.udpSendPacketLen * 2, 4096);
|
|
auto curBuf = largeBuf.get();
|
|
do {
|
|
curBuf->append(curBuf->capacity());
|
|
curBuf = curBuf->next();
|
|
} while (curBuf != largeBuf.get());
|
|
auto chainLen = largeBuf->computeChainDataLength();
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream1), std::move(largeBuf), false);
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream2),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream3),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
// Force the wraparound initially.
|
|
conn.schedulingState.nextScheduledStream = stream3 + 8;
|
|
scheduler.writeStreams(builder);
|
|
EXPECT_EQ(conn.schedulingState.nextScheduledStream, 4);
|
|
|
|
// Should write frames for stream2, stream3, followed by stream1 again.
|
|
NiceMock<MockQuicPacketBuilder> builder2;
|
|
EXPECT_CALL(builder2, remainingSpaceInPkt()).WillRepeatedly(Return(4096));
|
|
EXPECT_CALL(builder2, appendFrame(_)).WillRepeatedly(Invoke([&](auto f) {
|
|
builder2.frames_.push_back(f);
|
|
}));
|
|
scheduler.writeStreams(builder2);
|
|
auto& frames = builder2.frames_;
|
|
ASSERT_EQ(frames.size(), 3);
|
|
WriteStreamFrame f1(stream2, 0, 9, false);
|
|
WriteStreamFrame f2(stream3, 0, 9, false);
|
|
WriteStreamFrame f3(stream1, 0, chainLen, false);
|
|
ASSERT_TRUE(frames[0].asWriteStreamFrame());
|
|
EXPECT_EQ(*frames[0].asWriteStreamFrame(), f1);
|
|
ASSERT_TRUE(frames[1].asWriteStreamFrame());
|
|
EXPECT_EQ(*frames[1].asWriteStreamFrame(), f2);
|
|
ASSERT_TRUE(frames[2].asWriteStreamFrame());
|
|
EXPECT_EQ(*frames[2].asWriteStreamFrame(), f3);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, StreamFrameSchedulerRoundRobinStreamPerPacket) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.streamManager->setMaxLocalBidirectionalStreams(10);
|
|
conn.flowControlState.peerAdvertisedMaxOffset = 100000;
|
|
conn.flowControlState.peerAdvertisedInitialMaxStreamOffsetBidiRemote = 100000;
|
|
conn.transportSettings.streamFramePerPacket = true;
|
|
auto connId = getTestConnectionId();
|
|
StreamFrameScheduler scheduler(conn);
|
|
ShortHeader shortHeader1(
|
|
ProtectionType::KeyPhaseZero,
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder1(
|
|
conn.udpSendPacketLen,
|
|
std::move(shortHeader1),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
auto stream1 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto stream2 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto stream3 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto largeBuf = folly::IOBuf::createChain(conn.udpSendPacketLen * 2, 4096);
|
|
auto curBuf = largeBuf.get();
|
|
do {
|
|
curBuf->append(curBuf->capacity());
|
|
curBuf = curBuf->next();
|
|
} while (curBuf != largeBuf.get());
|
|
auto chainLen = largeBuf->computeChainDataLength();
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream1), std::move(largeBuf), false);
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream2),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream3),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
// Force the wraparound initially.
|
|
conn.schedulingState.nextScheduledStream = stream3 + 8;
|
|
scheduler.writeStreams(builder1);
|
|
EXPECT_EQ(conn.schedulingState.nextScheduledStream, 4);
|
|
|
|
// Should write frames for stream2, stream3, followed by stream1 again.
|
|
NiceMock<MockQuicPacketBuilder> builder2;
|
|
EXPECT_CALL(builder2, remainingSpaceInPkt()).WillRepeatedly(Return(4096));
|
|
EXPECT_CALL(builder2, appendFrame(_)).WillRepeatedly(Invoke([&](auto f) {
|
|
builder2.frames_.push_back(f);
|
|
}));
|
|
auto& frames = builder2.frames_;
|
|
scheduler.writeStreams(builder2);
|
|
ASSERT_EQ(frames.size(), 1);
|
|
scheduler.writeStreams(builder2);
|
|
ASSERT_EQ(frames.size(), 2);
|
|
scheduler.writeStreams(builder2);
|
|
ASSERT_EQ(frames.size(), 3);
|
|
WriteStreamFrame f1(stream2, 0, 9, false);
|
|
WriteStreamFrame f2(stream3, 0, 9, false);
|
|
WriteStreamFrame f3(stream1, 0, chainLen, false);
|
|
ASSERT_TRUE(frames[0].asWriteStreamFrame());
|
|
EXPECT_EQ(*frames[0].asWriteStreamFrame(), f1);
|
|
ASSERT_TRUE(frames[1].asWriteStreamFrame());
|
|
EXPECT_EQ(*frames[1].asWriteStreamFrame(), f2);
|
|
ASSERT_TRUE(frames[2].asWriteStreamFrame());
|
|
EXPECT_EQ(*frames[2].asWriteStreamFrame(), f3);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, StreamFrameSchedulerRoundRobinControl) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.streamManager->setMaxLocalBidirectionalStreams(10);
|
|
conn.flowControlState.peerAdvertisedMaxOffset = 100000;
|
|
conn.flowControlState.peerAdvertisedInitialMaxStreamOffsetBidiRemote = 100000;
|
|
auto connId = getTestConnectionId();
|
|
StreamFrameScheduler scheduler(conn);
|
|
ShortHeader shortHeader1(
|
|
ProtectionType::KeyPhaseZero,
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(shortHeader1),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
builder.encodePacketHeader();
|
|
auto stream1 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto stream2 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto stream3 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto stream4 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
conn.streamManager->setStreamAsControl(
|
|
*conn.streamManager->findStream(stream2));
|
|
conn.streamManager->setStreamAsControl(
|
|
*conn.streamManager->findStream(stream4));
|
|
auto largeBuf = folly::IOBuf::createChain(conn.udpSendPacketLen * 2, 4096);
|
|
auto curBuf = largeBuf.get();
|
|
do {
|
|
curBuf->append(curBuf->capacity());
|
|
curBuf = curBuf->next();
|
|
} while (curBuf != largeBuf.get());
|
|
auto chainLen = largeBuf->computeChainDataLength();
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream1), std::move(largeBuf), false);
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream2),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream3),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream4),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
// Force the wraparound initially.
|
|
conn.schedulingState.nextScheduledStream = stream4 + 8;
|
|
scheduler.writeStreams(builder);
|
|
EXPECT_EQ(conn.schedulingState.nextScheduledStream, stream3);
|
|
EXPECT_EQ(conn.schedulingState.nextScheduledControlStream, stream2);
|
|
|
|
// Should write frames for stream2, stream4, followed by stream 3 then 1.
|
|
NiceMock<MockQuicPacketBuilder> builder2;
|
|
EXPECT_CALL(builder2, remainingSpaceInPkt()).WillRepeatedly(Return(4096));
|
|
EXPECT_CALL(builder2, appendFrame(_)).WillRepeatedly(Invoke([&](auto f) {
|
|
builder2.frames_.push_back(f);
|
|
}));
|
|
scheduler.writeStreams(builder2);
|
|
auto& frames = builder2.frames_;
|
|
ASSERT_EQ(frames.size(), 4);
|
|
WriteStreamFrame f1(stream2, 0, 9, false);
|
|
WriteStreamFrame f2(stream4, 0, 9, false);
|
|
WriteStreamFrame f3(stream3, 0, 9, false);
|
|
WriteStreamFrame f4(stream1, 0, chainLen, false);
|
|
ASSERT_TRUE(frames[0].asWriteStreamFrame());
|
|
EXPECT_EQ(*frames[0].asWriteStreamFrame(), f1);
|
|
ASSERT_TRUE(frames[1].asWriteStreamFrame());
|
|
EXPECT_EQ(*frames[1].asWriteStreamFrame(), f2);
|
|
ASSERT_TRUE(frames[2].asWriteStreamFrame());
|
|
EXPECT_EQ(*frames[2].asWriteStreamFrame(), f3);
|
|
ASSERT_TRUE(frames[3].asWriteStreamFrame());
|
|
EXPECT_EQ(*frames[3].asWriteStreamFrame(), f4);
|
|
|
|
EXPECT_EQ(conn.schedulingState.nextScheduledStream, stream3);
|
|
EXPECT_EQ(conn.schedulingState.nextScheduledControlStream, stream2);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, StreamFrameSchedulerOneStream) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.streamManager->setMaxLocalBidirectionalStreams(10);
|
|
conn.flowControlState.peerAdvertisedMaxOffset = 100000;
|
|
conn.flowControlState.peerAdvertisedInitialMaxStreamOffsetBidiRemote = 100000;
|
|
auto connId = getTestConnectionId();
|
|
StreamFrameScheduler scheduler(conn);
|
|
ShortHeader shortHeader(
|
|
ProtectionType::KeyPhaseZero,
|
|
connId,
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
RegularQuicPacketBuilder builder(
|
|
conn.udpSendPacketLen,
|
|
std::move(shortHeader),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
builder.encodePacketHeader();
|
|
auto stream1 = conn.streamManager->createNextBidirectionalStream().value();
|
|
writeDataToQuicStream(*stream1, folly::IOBuf::copyBuffer("some data"), false);
|
|
scheduler.writeStreams(builder);
|
|
EXPECT_EQ(conn.schedulingState.nextScheduledStream, 0);
|
|
}
|
|
|
|
TEST_F(QuicPacketSchedulerTest, StreamFrameSchedulerRemoveOne) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.streamManager->setMaxLocalBidirectionalStreams(10);
|
|
conn.flowControlState.peerAdvertisedMaxOffset = 100000;
|
|
conn.flowControlState.peerAdvertisedInitialMaxStreamOffsetBidiRemote = 100000;
|
|
StreamFrameScheduler scheduler(conn);
|
|
NiceMock<MockQuicPacketBuilder> builder;
|
|
auto stream1 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
auto stream2 =
|
|
conn.streamManager->createNextBidirectionalStream().value()->id;
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream1),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
writeDataToQuicStream(
|
|
*conn.streamManager->findStream(stream2),
|
|
folly::IOBuf::copyBuffer("some data"),
|
|
false);
|
|
EXPECT_CALL(builder, remainingSpaceInPkt()).WillRepeatedly(Return(4096));
|
|
EXPECT_CALL(builder, appendFrame(_)).WillRepeatedly(Invoke([&](auto f) {
|
|
builder.frames_.push_back(f);
|
|
}));
|
|
scheduler.writeStreams(builder);
|
|
WriteStreamFrame f1(stream1, 0, 9, false);
|
|
WriteStreamFrame f2(stream2, 0, 9, false);
|
|
ASSERT_TRUE(builder.frames_[0].asWriteStreamFrame());
|
|
EXPECT_EQ(*builder.frames_[0].asWriteStreamFrame(), f1);
|
|
ASSERT_TRUE(builder.frames_[1].asWriteStreamFrame());
|
|
EXPECT_EQ(*builder.frames_[1].asWriteStreamFrame(), f2);
|
|
|
|
// Manually remove a stream and set the next scheduled to that stream.
|
|
builder.frames_.clear();
|
|
conn.streamManager->removeWritable(*conn.streamManager->findStream(stream2));
|
|
conn.schedulingState.nextScheduledStream = stream2;
|
|
scheduler.writeStreams(builder);
|
|
ASSERT_EQ(builder.frames_.size(), 1);
|
|
ASSERT_TRUE(builder.frames_[0].asWriteStreamFrame());
|
|
EXPECT_EQ(*builder.frames_[0].asWriteStreamFrame(), f1);
|
|
}
|
|
|
|
TEST_F(
|
|
QuicPacketSchedulerTest,
|
|
CloningSchedulerWithInplaceBuilderDoNotEncodeHeaderWithoutBuild) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.transportSettings.dataPathType = DataPathType::ContinuousMemory;
|
|
SimpleBufAccessor bufAccessor(2000);
|
|
auto buf = bufAccessor.obtain();
|
|
EXPECT_EQ(buf->length(), 0);
|
|
bufAccessor.release(std::move(buf));
|
|
conn.bufAccessor = &bufAccessor;
|
|
|
|
FrameScheduler noopScheduler("frame");
|
|
ASSERT_FALSE(noopScheduler.hasData());
|
|
CloningScheduler cloningScheduler(noopScheduler, conn, "Little Hurry", 0);
|
|
addOutstandingPacket(conn);
|
|
// There needs to have retransmittable frame for the rebuilder to work
|
|
conn.outstandingPackets.back().packet.frames.push_back(
|
|
MaxDataFrame(conn.flowControlState.advertisedMaxOffset));
|
|
// Lie about the encodedSize to let the Cloner skip it:
|
|
conn.outstandingPackets.back().encodedSize = kDefaultUDPSendPacketLen * 2;
|
|
EXPECT_TRUE(cloningScheduler.hasData());
|
|
|
|
ASSERT_FALSE(noopScheduler.hasData());
|
|
ShortHeader header(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
InplaceQuicPacketBuilder builder(
|
|
bufAccessor,
|
|
conn.udpSendPacketLen,
|
|
std::move(header),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
auto result = cloningScheduler.scheduleFramesForPacket(
|
|
std::move(builder), kDefaultUDPSendPacketLen);
|
|
EXPECT_FALSE(result.packetEvent.has_value());
|
|
|
|
// Nothing was written into the buffer:
|
|
EXPECT_TRUE(bufAccessor.ownsBuffer());
|
|
buf = bufAccessor.obtain();
|
|
EXPECT_EQ(buf->length(), 0);
|
|
}
|
|
|
|
TEST_F(
|
|
QuicPacketSchedulerTest,
|
|
CloningSchedulerWithInplaceBuilderRollbackBufWhenFailToRebuild) {
|
|
QuicClientConnectionState conn(
|
|
FizzClientQuicHandshakeContext::Builder().build());
|
|
conn.transportSettings.dataPathType = DataPathType::ContinuousMemory;
|
|
SimpleBufAccessor bufAccessor(2000);
|
|
auto buf = bufAccessor.obtain();
|
|
EXPECT_EQ(buf->length(), 0);
|
|
bufAccessor.release(std::move(buf));
|
|
conn.bufAccessor = &bufAccessor;
|
|
|
|
FrameScheduler noopScheduler("frame");
|
|
ASSERT_FALSE(noopScheduler.hasData());
|
|
CloningScheduler cloningScheduler(noopScheduler, conn, "HotPot", 0);
|
|
addOutstandingPacket(conn);
|
|
// Not adding frame to this outstanding packet so that rebuild will fail:
|
|
ASSERT_TRUE(conn.outstandingPackets.back().packet.frames.empty());
|
|
EXPECT_TRUE(cloningScheduler.hasData());
|
|
|
|
ASSERT_FALSE(noopScheduler.hasData());
|
|
ShortHeader header(
|
|
ProtectionType::KeyPhaseOne,
|
|
conn.clientConnectionId.value_or(getTestConnectionId()),
|
|
getNextPacketNum(conn, PacketNumberSpace::AppData));
|
|
InplaceQuicPacketBuilder builder(
|
|
bufAccessor,
|
|
conn.udpSendPacketLen,
|
|
std::move(header),
|
|
conn.ackStates.appDataAckState.largestAckedByPeer);
|
|
auto result = cloningScheduler.scheduleFramesForPacket(
|
|
std::move(builder), kDefaultUDPSendPacketLen);
|
|
EXPECT_FALSE(result.packetEvent.has_value());
|
|
|
|
// Nothing was written into the buffer:
|
|
EXPECT_TRUE(bufAccessor.ownsBuffer());
|
|
buf = bufAccessor.obtain();
|
|
EXPECT_EQ(buf->length(), 0);
|
|
}
|
|
|
|
} // namespace test
|
|
} // namespace quic
|