mirror of
https://github.com/DrKLO/Telegram.git
synced 2025-01-10 03:52:40 +01:00
346 lines
14 KiB
C
346 lines
14 KiB
C
|
/*
|
||
|
* Copyright (c) 2021 The WebRTC project authors. All Rights Reserved.
|
||
|
*
|
||
|
* Use of this source code is governed by a BSD-style license
|
||
|
* that can be found in the LICENSE file in the root of the source
|
||
|
* tree. An additional intellectual property rights grant can be found
|
||
|
* in the file PATENTS. All contributing project authors may
|
||
|
* be found in the AUTHORS file in the root of the source tree.
|
||
|
*/
|
||
|
#ifndef NET_DCSCTP_TX_RETRANSMISSION_QUEUE_H_
|
||
|
#define NET_DCSCTP_TX_RETRANSMISSION_QUEUE_H_
|
||
|
|
||
|
#include <cstdint>
|
||
|
#include <functional>
|
||
|
#include <map>
|
||
|
#include <string>
|
||
|
#include <utility>
|
||
|
#include <vector>
|
||
|
|
||
|
#include "absl/strings/string_view.h"
|
||
|
#include "absl/types/optional.h"
|
||
|
#include "api/array_view.h"
|
||
|
#include "net/dcsctp/common/sequence_numbers.h"
|
||
|
#include "net/dcsctp/packet/chunk/forward_tsn_chunk.h"
|
||
|
#include "net/dcsctp/packet/chunk/iforward_tsn_chunk.h"
|
||
|
#include "net/dcsctp/packet/chunk/sack_chunk.h"
|
||
|
#include "net/dcsctp/packet/data.h"
|
||
|
#include "net/dcsctp/public/dcsctp_options.h"
|
||
|
#include "net/dcsctp/timer/timer.h"
|
||
|
#include "net/dcsctp/tx/retransmission_timeout.h"
|
||
|
#include "net/dcsctp/tx/send_queue.h"
|
||
|
|
||
|
namespace dcsctp {
|
||
|
|
||
|
// The RetransmissionQueue manages all DATA/I-DATA chunks that are in-flight and
|
||
|
// schedules them to be retransmitted if necessary. Chunks are retransmitted
|
||
|
// when they have been lost for a number of consecutive SACKs, or when the
|
||
|
// retransmission timer, `t3_rtx` expires.
|
||
|
//
|
||
|
// As congestion control is tightly connected with the state of transmitted
|
||
|
// packets, that's also managed here to limit the amount of data that is
|
||
|
// in-flight (sent, but not yet acknowledged).
|
||
|
class RetransmissionQueue {
|
||
|
public:
|
||
|
static constexpr size_t kMinimumFragmentedPayload = 10;
|
||
|
// State for DATA chunks (message fragments) in the queue.
|
||
|
enum class State {
|
||
|
// The chunk has been sent but not received yet (from the sender's point of
|
||
|
// view, as no SACK has been received yet that reference this chunk).
|
||
|
kInFlight,
|
||
|
// A SACK has been received which explicitly marked this chunk as missing -
|
||
|
// it's now NACKED and may be retransmitted if NACKED enough times.
|
||
|
kNacked,
|
||
|
// A chunk that will be retransmitted when possible.
|
||
|
kToBeRetransmitted,
|
||
|
// A SACK has been received which explicitly marked this chunk as received.
|
||
|
kAcked,
|
||
|
// A chunk whose message has expired or has been retransmitted too many
|
||
|
// times (RFC3758). It will not be retransmitted anymore.
|
||
|
kAbandoned,
|
||
|
};
|
||
|
|
||
|
// Creates a RetransmissionQueue which will send data using `initial_tsn` as
|
||
|
// the first TSN to use for sent fragments. It will poll data from
|
||
|
// `send_queue` and call `on_send_queue_empty` when it is empty. When
|
||
|
// SACKs are received, it will estimate the RTT, and call `on_new_rtt`. When
|
||
|
// an outstanding chunk has been ACKed, it will call
|
||
|
// `on_clear_retransmission_counter` and will also use `t3_rtx`, which is the
|
||
|
// SCTP retransmission timer to manage retransmissions.
|
||
|
RetransmissionQueue(absl::string_view log_prefix,
|
||
|
TSN initial_tsn,
|
||
|
size_t a_rwnd,
|
||
|
SendQueue& send_queue,
|
||
|
std::function<void(DurationMs rtt)> on_new_rtt,
|
||
|
std::function<void()> on_send_queue_empty,
|
||
|
std::function<void()> on_clear_retransmission_counter,
|
||
|
Timer& t3_rtx,
|
||
|
const DcSctpOptions& options,
|
||
|
bool supports_partial_reliability = true,
|
||
|
bool use_message_interleaving = false);
|
||
|
|
||
|
// Handles a received SACK. Returns true if the `sack` was processed and
|
||
|
// false if it was discarded due to received out-of-order and not relevant.
|
||
|
bool HandleSack(TimeMs now, const SackChunk& sack);
|
||
|
|
||
|
// Handles an expired retransmission timer.
|
||
|
void HandleT3RtxTimerExpiry();
|
||
|
|
||
|
// Returns a list of chunks to send that would fit in one SCTP packet with
|
||
|
// `bytes_remaining_in_packet` bytes available. This may be further limited by
|
||
|
// the congestion control windows. Note that `ShouldSendForwardTSN` must be
|
||
|
// called prior to this method, to abandon expired chunks, as this method will
|
||
|
// not expire any chunks.
|
||
|
std::vector<std::pair<TSN, Data>> GetChunksToSend(
|
||
|
TimeMs now,
|
||
|
size_t bytes_remaining_in_packet);
|
||
|
|
||
|
// Returns the internal state of all queued chunks. This is only used in
|
||
|
// unit-tests.
|
||
|
std::vector<std::pair<TSN, State>> GetChunkStatesForTesting() const;
|
||
|
|
||
|
// Returns the next TSN that will be allocated for sent DATA chunks.
|
||
|
TSN next_tsn() const { return next_tsn_.Wrap(); }
|
||
|
|
||
|
// Returns the size of the congestion window, in bytes. This is the number of
|
||
|
// bytes that may be in-flight.
|
||
|
size_t cwnd() const { return cwnd_; }
|
||
|
|
||
|
// Overrides the current congestion window size.
|
||
|
void set_cwnd(size_t cwnd) { cwnd_ = cwnd; }
|
||
|
|
||
|
// Returns the current receiver window size.
|
||
|
size_t rwnd() const { return rwnd_; }
|
||
|
|
||
|
// Returns the number of bytes of packets that are in-flight.
|
||
|
size_t outstanding_bytes() const { return outstanding_bytes_; }
|
||
|
|
||
|
// Given the current time `now`, it will evaluate if there are chunks that
|
||
|
// have expired and that need to be discarded. It returns true if a
|
||
|
// FORWARD-TSN should be sent.
|
||
|
bool ShouldSendForwardTsn(TimeMs now);
|
||
|
|
||
|
// Creates a FORWARD-TSN chunk.
|
||
|
ForwardTsnChunk CreateForwardTsn() const;
|
||
|
|
||
|
// Creates an I-FORWARD-TSN chunk.
|
||
|
IForwardTsnChunk CreateIForwardTsn() const;
|
||
|
|
||
|
// See the SendQueue for a longer description of these methods related
|
||
|
// to stream resetting.
|
||
|
void PrepareResetStreams(rtc::ArrayView<const StreamID> streams);
|
||
|
bool CanResetStreams() const;
|
||
|
void CommitResetStreams();
|
||
|
void RollbackResetStreams();
|
||
|
|
||
|
private:
|
||
|
enum class CongestionAlgorithmPhase {
|
||
|
kSlowStart,
|
||
|
kCongestionAvoidance,
|
||
|
};
|
||
|
|
||
|
// A fragmented message's DATA chunk while in the retransmission queue, and
|
||
|
// its associated metadata.
|
||
|
class TxData {
|
||
|
public:
|
||
|
explicit TxData(Data data,
|
||
|
absl::optional<size_t> max_retransmissions,
|
||
|
TimeMs time_sent,
|
||
|
absl::optional<TimeMs> expires_at)
|
||
|
: max_retransmissions_(max_retransmissions),
|
||
|
time_sent_(time_sent),
|
||
|
expires_at_(expires_at),
|
||
|
data_(std::move(data)) {}
|
||
|
|
||
|
TimeMs time_sent() const { return time_sent_; }
|
||
|
|
||
|
State state() const { return state_; }
|
||
|
void SetState(State state) { state_ = state; }
|
||
|
|
||
|
const Data& data() const { return data_; }
|
||
|
|
||
|
// Nacks an item. If it has been nacked enough times, it will be marked for
|
||
|
// retransmission.
|
||
|
void Nack();
|
||
|
void Retransmit();
|
||
|
|
||
|
bool has_been_retransmitted() { return num_retransmissions_ > 0; }
|
||
|
|
||
|
// Given the current time, and the current state of this DATA chunk, it will
|
||
|
// indicate if it has expired (SCTP Partial Reliability Extension).
|
||
|
bool has_expired(TimeMs now) const;
|
||
|
|
||
|
private:
|
||
|
State state_ = State::kInFlight;
|
||
|
// The number of times the DATA chunk has been nacked (by having received a
|
||
|
// SACK which doesn't include it). Will be cleared on retransmissions.
|
||
|
size_t nack_count_ = 0;
|
||
|
// The number of times the DATA chunk has been retransmitted.
|
||
|
size_t num_retransmissions_ = 0;
|
||
|
// If the message was sent with a maximum number of retransmissions, this is
|
||
|
// set to that number. The value zero (0) means that it will never be
|
||
|
// retransmitted.
|
||
|
const absl::optional<size_t> max_retransmissions_;
|
||
|
// When the packet was sent, and placed in this queue.
|
||
|
const TimeMs time_sent_;
|
||
|
// If the message was sent with an expiration time, this is set.
|
||
|
const absl::optional<TimeMs> expires_at_;
|
||
|
// The actual data to send/retransmit.
|
||
|
Data data_;
|
||
|
};
|
||
|
|
||
|
// Contains variables scoped to a processing of an incoming SACK.
|
||
|
struct AckInfo {
|
||
|
explicit AckInfo(UnwrappedTSN cumulative_tsn_ack)
|
||
|
: highest_tsn_acked(cumulative_tsn_ack) {}
|
||
|
|
||
|
// All TSNs that have been acked (for the first time) in this SACK.
|
||
|
std::vector<TSN> acked_tsns;
|
||
|
|
||
|
// Bytes acked by increasing cumulative_tsn_ack in this SACK
|
||
|
size_t bytes_acked_by_cumulative_tsn_ack = 0;
|
||
|
|
||
|
// Bytes acked by gap blocks in this SACK.
|
||
|
size_t bytes_acked_by_new_gap_ack_blocks = 0;
|
||
|
|
||
|
// Indicates if this SACK indicates that packet loss has occurred. Just
|
||
|
// because a packet is missing in the SACK doesn't necessarily mean that
|
||
|
// there is packet loss as that packet might be in-flight and received
|
||
|
// out-of-order. But when it has been reported missing consecutive times, it
|
||
|
// will eventually be considered "lost" and this will be set.
|
||
|
bool has_packet_loss = false;
|
||
|
|
||
|
// Highest TSN Newly Acknowledged, an SCTP variable.
|
||
|
UnwrappedTSN highest_tsn_acked;
|
||
|
};
|
||
|
|
||
|
// Returns how large a chunk will be, serialized, carrying the data
|
||
|
size_t GetSerializedChunkSize(const Data& data) const;
|
||
|
|
||
|
// Indicates if the congestion control algorithm is in "fast recovery".
|
||
|
bool is_in_fast_recovery() const {
|
||
|
return fast_recovery_exit_tsn_.has_value();
|
||
|
}
|
||
|
|
||
|
// Indicates if the congestion control algorithm is in "fast retransmit".
|
||
|
bool is_in_fast_retransmit() const { return is_in_fast_retransmit_; }
|
||
|
|
||
|
// Indicates if the provided SACK is valid given what has previously been
|
||
|
// received. If it returns false, the SACK is most likely a duplicate of
|
||
|
// something already seen, so this returning false doesn't necessarily mean
|
||
|
// that the SACK is illegal.
|
||
|
bool IsSackValid(const SackChunk& sack) const;
|
||
|
|
||
|
// Given a `cumulative_tsn_ack` from an incoming SACK, will remove those items
|
||
|
// in the retransmission queue up until this value and will update `ack_info`
|
||
|
// by setting `bytes_acked_by_cumulative_tsn_ack` and `acked_tsns`.
|
||
|
void RemoveAcked(UnwrappedTSN cumulative_tsn_ack, AckInfo& ack_info);
|
||
|
|
||
|
// Will mark the chunks covered by the `gap_ack_blocks` from an incoming SACK
|
||
|
// as "acked" and update `ack_info` by adding new TSNs to `added_tsns`.
|
||
|
void AckGapBlocks(UnwrappedTSN cumulative_tsn_ack,
|
||
|
rtc::ArrayView<const SackChunk::GapAckBlock> gap_ack_blocks,
|
||
|
AckInfo& ack_info);
|
||
|
|
||
|
// Mark chunks reported as "missing", as "nacked" or "to be retransmitted"
|
||
|
// depending how many times this has happened. Only packets up until
|
||
|
// `ack_info.highest_tsn_acked` (highest TSN newly acknowledged) are
|
||
|
// nacked/retransmitted. The method will set `ack_info.has_packet_loss`.
|
||
|
void NackBetweenAckBlocks(
|
||
|
UnwrappedTSN cumulative_tsn_ack,
|
||
|
rtc::ArrayView<const SackChunk::GapAckBlock> gap_ack_blocks,
|
||
|
AckInfo& ack_info);
|
||
|
|
||
|
// When a SACK chunk is received, this method will be called which _may_ call
|
||
|
// into the `RetransmissionTimeout` to update the RTO.
|
||
|
void UpdateRTT(TimeMs now, UnwrappedTSN cumulative_tsn_ack);
|
||
|
|
||
|
// If the congestion control is in "fast recovery mode", this may be exited
|
||
|
// now.
|
||
|
void MaybeExitFastRecovery(UnwrappedTSN cumulative_tsn_ack);
|
||
|
|
||
|
// If chunks have been ACKed, stop the retransmission timer.
|
||
|
void StopT3RtxTimerOnIncreasedCumulativeTsnAck(
|
||
|
UnwrappedTSN cumulative_tsn_ack);
|
||
|
|
||
|
// Update the congestion control algorithm given as the cumulative ack TSN
|
||
|
// value has increased, as reported in an incoming SACK chunk.
|
||
|
void HandleIncreasedCumulativeTsnAck(size_t outstanding_bytes,
|
||
|
size_t total_bytes_acked);
|
||
|
// Update the congestion control algorithm, given as packet loss has been
|
||
|
// detected, as reported in an incoming SACK chunk.
|
||
|
void HandlePacketLoss(UnwrappedTSN highest_tsn_acked);
|
||
|
// Recalculate the number of in-flight payload bytes.
|
||
|
void RecalculateOutstandingBytes();
|
||
|
// Update the view of the receiver window size.
|
||
|
void UpdateReceiverWindow(uint32_t a_rwnd);
|
||
|
// Given `max_size` of space left in a packet, which chunks can be added to
|
||
|
// it?
|
||
|
std::vector<std::pair<TSN, Data>> GetChunksToBeRetransmitted(size_t max_size);
|
||
|
// If there is data sent and not ACKED, ensure that the retransmission timer
|
||
|
// is running.
|
||
|
void StartT3RtxTimerIfOutstandingData();
|
||
|
|
||
|
// Given the current time `now_ms`, expire chunks that have a limited
|
||
|
// lifetime.
|
||
|
void ExpireChunks(TimeMs now);
|
||
|
// Given that a message fragment, `item` has expired, expire all other
|
||
|
// fragments that share the same message - even never-before-sent fragments
|
||
|
// that are still in the SendQueue.
|
||
|
void ExpireAllFor(const RetransmissionQueue::TxData& item);
|
||
|
|
||
|
// Returns the current congestion control algorithm phase.
|
||
|
CongestionAlgorithmPhase phase() const {
|
||
|
return (cwnd_ <= ssthresh_)
|
||
|
? CongestionAlgorithmPhase::kSlowStart
|
||
|
: CongestionAlgorithmPhase::kCongestionAvoidance;
|
||
|
}
|
||
|
|
||
|
const DcSctpOptions options_;
|
||
|
// If the peer supports RFC3758 - SCTP Partial Reliability Extension.
|
||
|
const bool partial_reliability_;
|
||
|
const std::string log_prefix_;
|
||
|
// The size of the data chunk (DATA/I-DATA) header that is used.
|
||
|
const size_t data_chunk_header_size_;
|
||
|
// Called when a new RTT measurement has been done
|
||
|
const std::function<void(DurationMs rtt)> on_new_rtt_;
|
||
|
// Called when the send queue is empty.
|
||
|
const std::function<void()> on_send_queue_empty_;
|
||
|
// Called when a SACK has been seen that cleared the retransmission counter.
|
||
|
const std::function<void()> on_clear_retransmission_counter_;
|
||
|
// The retransmission counter.
|
||
|
Timer& t3_rtx_;
|
||
|
// Unwraps TSNs
|
||
|
UnwrappedTSN::Unwrapper tsn_unwrapper_;
|
||
|
|
||
|
// Congestion Window. Number of bytes that may be in-flight (sent, not acked).
|
||
|
size_t cwnd_;
|
||
|
// Receive Window. Number of bytes available in the receiver's RX buffer.
|
||
|
size_t rwnd_;
|
||
|
// Slow Start Threshold. See RFC4960.
|
||
|
size_t ssthresh_;
|
||
|
// Partial Bytes Acked. See RFC4960.
|
||
|
size_t partial_bytes_acked_ = 0;
|
||
|
// If set, fast recovery is enabled until this TSN has been cumulative
|
||
|
// acked.
|
||
|
absl::optional<UnwrappedTSN> fast_recovery_exit_tsn_ = absl::nullopt;
|
||
|
// Indicates if the congestion algorithm is in fast retransmit.
|
||
|
bool is_in_fast_retransmit_ = false;
|
||
|
|
||
|
// Next TSN to used.
|
||
|
UnwrappedTSN next_tsn_;
|
||
|
// The last cumulative TSN ack number
|
||
|
UnwrappedTSN last_cumulative_tsn_ack_;
|
||
|
// The send queue.
|
||
|
SendQueue& send_queue_;
|
||
|
// All the outstanding data chunks that are in-flight and that have not been
|
||
|
// cumulative acked. Note that it also contains chunks that have been acked in
|
||
|
// gap ack blocks.
|
||
|
std::map<UnwrappedTSN, TxData> outstanding_data_;
|
||
|
// The sum of the message bytes of the send_queue_
|
||
|
size_t outstanding_bytes_ = 0;
|
||
|
};
|
||
|
} // namespace dcsctp
|
||
|
|
||
|
#endif // NET_DCSCTP_TX_RETRANSMISSION_QUEUE_H_
|