1
0
mirror of https://github.com/facebookincubator/mvfst.git synced 2025-11-27 03:41:14 +03:00
Files
mvfst/quic/loss/QuicLossFunctions.cpp
Yang Chi dc27ebf05d Rename ackFrameMatchesRetransmitBuffer to streamFrameMatchesRetransmitBuffer
Summary:
The function indeed compares a Stream frame to a buffer, not an Ack
frame to a buffer.

Reviewed By: sharma95

Differential Revision: D18000632

fbshipit-source-id: fbbc36378404f8f52a199114af48b9020d0569e8
2019-10-18 15:54:26 -07:00

206 lines
7.2 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/loss/QuicLossFunctions.h"
namespace quic {
std::chrono::microseconds calculatePTO(const QuicConnectionStateBase& conn) {
return conn.lossState.srtt + 4 * conn.lossState.rttvar +
conn.lossState.maxAckDelay;
}
bool isPersistentCongestion(
const QuicConnectionStateBase& conn,
TimePoint lostPeriodStart,
TimePoint lostPeriodEnd) noexcept {
if (conn.lossState.srtt == 0us) {
return false;
}
auto pto = calculatePTO(conn);
return (lostPeriodEnd - lostPeriodStart) >=
pto * kPersistentCongestionThreshold;
}
void onPTOAlarm(QuicConnectionStateBase& conn) {
VLOG(10) << __func__ << " " << conn;
QUIC_TRACE(
pto_alarm,
conn,
conn.lossState.largestSent,
conn.lossState.ptoCount,
(uint64_t)conn.outstandingPackets.size());
QUIC_STATS(conn.infoCallback, onPTO);
conn.lossState.ptoCount++;
conn.lossState.totalPTOCount++;
if (conn.qLogger) {
conn.qLogger->addLossAlarm(
conn.lossState.largestSent,
conn.lossState.ptoCount,
(uint64_t)conn.outstandingPackets.size(),
kPtoAlarm);
}
if (conn.lossState.ptoCount == conn.transportSettings.maxNumPTOs) {
throw QuicInternalException("Exceeded max PTO", LocalErrorCode::NO_ERROR);
}
conn.pendingEvents.numProbePackets = kPacketToSendForPTO;
}
void markPacketLoss(
QuicConnectionStateBase& conn,
RegularQuicWritePacket& packet,
bool processed,
PacketNum currentPacketNum) {
for (auto& packetFrame : packet.frames) {
switch (packetFrame.type()) {
case QuicWriteFrame::Type::MaxStreamDataFrame_E: {
MaxStreamDataFrame& frame = *packetFrame.asMaxStreamDataFrame();
// For all other frames, we process it if it's not from a clone
// packet, or if the clone and its siblings have never been processed.
// But for both MaxData and MaxStreamData, clone and its siblings may
// have different values. So we process it if it matches the
// latestMaxDataPacket or latestMaxStreamDataPacket. If an older
// packet also has such frames, it's ok to skip process of such loss
// since newer value is already sent in later packets.
auto stream = conn.streamManager->getStream(frame.streamId);
if (!stream) {
break;
}
// TODO: check for the stream is in Open or HalfClosedLocal state, the
// peer doesn't need a flow control update in these cases.
if (stream->latestMaxStreamDataPacket == currentPacketNum) {
onStreamWindowUpdateLost(*stream);
}
break;
}
case QuicWriteFrame::Type::MaxDataFrame_E: {
if (conn.latestMaxDataPacket == currentPacketNum) {
onConnWindowUpdateLost(conn);
}
break;
}
// For other frame types, we only process them if the packet is not a
// processed clone.
case QuicWriteFrame::Type::WriteStreamFrame_E: {
WriteStreamFrame frame = *packetFrame.asWriteStreamFrame();
if (processed) {
break;
}
auto stream = conn.streamManager->getStream(frame.streamId);
if (!stream) {
break;
}
auto bufferItr = std::lower_bound(
stream->retransmissionBuffer.begin(),
stream->retransmissionBuffer.end(),
frame.offset,
[](const auto& buffer, const auto& offset) {
return buffer.offset < offset;
});
if (bufferItr == stream->retransmissionBuffer.end()) {
// It's possible that the stream was reset or data on the stream was
// skipped while we discovered that its packet was lost so we might
// not have the offset.
break;
}
// The original rxmt offset might have been bumped up after it was
// shrunk due to egress partially reliable skip.
if (!streamFrameMatchesRetransmitBuffer(*stream, frame, *bufferItr)) {
break;
}
stream->lossBuffer.insert(
std::upper_bound(
stream->lossBuffer.begin(),
stream->lossBuffer.end(),
bufferItr->offset,
[](const auto& offset, const auto& buffer) {
return offset < buffer.offset;
}),
std::move(*bufferItr));
stream->retransmissionBuffer.erase(bufferItr);
conn.streamManager->updateLossStreams(*stream);
break;
}
case QuicWriteFrame::Type::WriteCryptoFrame_E: {
WriteCryptoFrame& frame = *packetFrame.asWriteCryptoFrame();
if (processed) {
break;
}
auto protectionType = packet.header.getProtectionType();
auto encryptionLevel = protectionTypeToEncryptionLevel(protectionType);
auto cryptoStream = getCryptoStream(*conn.cryptoState, encryptionLevel);
auto bufferItr = std::lower_bound(
cryptoStream->retransmissionBuffer.begin(),
cryptoStream->retransmissionBuffer.end(),
frame.offset,
[](const auto& buffer, const auto& offset) {
return buffer.offset < offset;
});
if (bufferItr == cryptoStream->retransmissionBuffer.end()) {
// It's possible that the stream was reset while we discovered that
// it's packet was lost so we might not have the offset.
break;
}
DCHECK_EQ(bufferItr->offset, frame.offset);
cryptoStream->lossBuffer.insert(
std::upper_bound(
cryptoStream->lossBuffer.begin(),
cryptoStream->lossBuffer.end(),
bufferItr->offset,
[](const auto& offset, const auto& buffer) {
return offset < buffer.offset;
}),
std::move(*bufferItr));
cryptoStream->retransmissionBuffer.erase(bufferItr);
break;
}
case QuicWriteFrame::Type::RstStreamFrame_E: {
RstStreamFrame& frame = *packetFrame.asRstStreamFrame();
if (processed) {
break;
}
auto stream = conn.streamManager->getStream(frame.streamId);
if (!stream) {
// If the stream is dead, ignore the retransmissions of the rst
// stream.
break;
}
// Add the lost RstStreamFrame back to pendingEvents:
conn.pendingEvents.resets.insert({frame.streamId, frame});
break;
}
case QuicWriteFrame::Type::StreamDataBlockedFrame_E: {
StreamDataBlockedFrame& frame = *packetFrame.asStreamDataBlockedFrame();
if (processed) {
break;
}
auto stream = conn.streamManager->getStream(frame.streamId);
// TODO: check for retransmittable
if (!stream) {
break;
}
onBlockedLost(*stream);
break;
}
case QuicWriteFrame::Type::QuicSimpleFrame_E: {
QuicSimpleFrame& frame = *packetFrame.asQuicSimpleFrame();
if (processed) {
break;
}
updateSimpleFrameOnPacketLoss(conn, frame);
break;
}
default:
// ignore the rest of the frames.
break;
}
}
}
} // namespace quic