Documentation ¶
Index ¶
- Constants
- Variables
- func MaxFilter(a, b int64) bool
- func MinFilter(a, b int64) bool
- func NewBBRSender(clock Clock, rttStats *RTTStats, ...) *bbrSender
- func NewCubicSender(clock Clock, rttStats *RTTStats, reno bool, ...) *cubicSender
- func SentPacketToSendTimeState(sentPacket *ConnectionStateOnSentPacket, sendTimeState *SendTimeState)
- type Bandwidth
- type BandwidthSample
- type BandwidthSampler
- func (s *BandwidthSampler) OnAppLimited()
- func (s *BandwidthSampler) OnPacketAcked(ackTime time.Time, lastAckedPacket protocol.PacketNumber) *BandwidthSample
- func (s *BandwidthSampler) OnPacketLost(packetNumber protocol.PacketNumber) SendTimeState
- func (s *BandwidthSampler) OnPacketSent(sentTime time.Time, lastSentPacket protocol.PacketNumber, ...)
- type Clock
- type CongestionEvent
- type ConnectionStateOnSentPacket
- type ConnectionStates
- func (s *ConnectionStates) Get(packetNumber protocol.PacketNumber) *ConnectionStateOnSentPacket
- func (s *ConnectionStates) Insert(packetNumber protocol.PacketNumber, sentTime time.Time, ...) bool
- func (s *ConnectionStates) Remove(packetNumber protocol.PacketNumber) (bool, *ConnectionStateOnSentPacket)
- type Cubic
- func (c *Cubic) CongestionWindowAfterAck(ackedBytes protocol.ByteCount, currentCongestionWindow protocol.ByteCount, ...) protocol.ByteCount
- func (c *Cubic) CongestionWindowAfterPacketLoss(currentCongestionWindow protocol.ByteCount) protocol.ByteCount
- func (c *Cubic) OnApplicationLimited()
- func (c *Cubic) Reset()
- func (c *Cubic) SetNumConnections(n int)
- type DefaultClock
- type HybridSlowStart
- func (s *HybridSlowStart) IsEndOfRound(ack protocol.PacketNumber) bool
- func (s *HybridSlowStart) OnPacketAcked(ackedPacketNumber protocol.PacketNumber)
- func (s *HybridSlowStart) OnPacketSent(packetNumber protocol.PacketNumber)
- func (s *HybridSlowStart) Restart()
- func (s *HybridSlowStart) ShouldExitSlowStart(latestRTT time.Duration, minRTT time.Duration, ...) bool
- func (s *HybridSlowStart) StartReceiveRound(lastSent protocol.PacketNumber)
- func (s *HybridSlowStart) Started() bool
- type PrrSender
- type RTTStats
- func (r *RTTStats) ExpireSmoothedMetrics()
- func (r *RTTStats) LatestRTT() time.Duration
- func (r *RTTStats) MeanDeviation() time.Duration
- func (r *RTTStats) MinRTT() time.Duration
- func (r *RTTStats) OnConnectionMigration()
- func (r *RTTStats) SmoothedOrInitialRTT() time.Duration
- func (r *RTTStats) SmoothedRTT() time.Duration
- func (r *RTTStats) UpdateRTT(sendDelta, ackDelay time.Duration, now time.Time)
- type Sample
- type SendAlgorithm
- type SendAlgorithmWithDebugInfos
- type SendTimeState
- type WindowedFilter
- func (f *WindowedFilter) GetBest() int64
- func (f *WindowedFilter) GetSecondBest() int64
- func (f *WindowedFilter) GetThirdBest() int64
- func (f *WindowedFilter) Reset(newSample int64, newTime int64)
- func (f *WindowedFilter) SetWindowLength(windowLength int64)
- func (f *WindowedFilter) Update(sample int64, time int64)
Constants ¶
const ( // Startup phase of the connection. STARTUP = iota // After achieving the highest possible bandwidth during the startup, lower // the pacing rate in order to drain the queue. DRAIN // Cruising mode. PROBE_BW // Temporarily slow down sending in order to empty the buffer and measure // the real minimum RTT. PROBE_RTT )
const ( // Do not limit. NOT_IN_RECOVERY = iota // Allow an extra outstanding byte for each byte acknowledged. CONSERVATION // Allow two extra outstanding bytes for each byte acknowledged (slow // start). GROWTH )
Variables ¶
var ( // The maximum outgoing packet size allowed. // The maximum packet size of any QUIC packet over IPv6, based on ethernet's max // size, minus the IP and UDP headers. IPv6 has a 40 byte header, UDP adds an // additional 8 bytes. This is a total overhead of 48 bytes. Ethernet's // max packet size is 1500 bytes, 1500 - 48 = 1452. MaxOutgoingPacketSize = protocol.ByteCount(1452) // Default maximum packet size used in the Linux TCP implementation. // Used in QUIC for congestion window computations in bytes. MaxSegmentSize = protocol.DefaultTCPMSS // Default initial rtt used before any samples are received. InitialRtt = 100 * time.Millisecond // Constants based on TCP defaults. // The minimum CWND to ensure delayed acks don't reduce bandwidth measurements. // Does not inflate the pacing rate. DefaultMinimumCongestionWindow = 4 * protocol.DefaultTCPMSS // The gain used for the STARTUP, equal to 2/ln(2). DefaultHighGain = 2.885 // The gain used in STARTUP after loss has been detected. // 1.5 is enough to allow for 25% exogenous loss and still observe a 25% growth // in measured bandwidth. StartupAfterLossGain = 1.5 // The cycle of gains used during the PROBE_BW stage. PacingGain = []float64{1.25, 0.75, 1, 1, 1, 1, 1, 1} // The length of the gain cycle. GainCycleLength = len(PacingGain) // The size of the bandwidth filter window, in round-trips. BandwidthWindowSize = GainCycleLength + 2 // The time after which the current min_rtt value expires. MinRttExpiry = 10 * time.Second // The minimum time the connection can spend in PROBE_RTT mode. ProbeRttTime = time.Millisecond * 200 // If the bandwidth does not increase by the factor of |kStartupGrowthTarget| // within |kRoundTripsWithoutGrowthBeforeExitingStartup| rounds, the connection // will exit the STARTUP mode. StartupGrowthTarget = 1.25 RoundTripsWithoutGrowthBeforeExitingStartup = int64(3) // Coefficient of target congestion window to use when basing PROBE_RTT on BDP. ModerateProbeRttMultiplier = 0.75 // Coefficient to determine if a new RTT is sufficiently similar to min_rtt that // we don't need to enter PROBE_RTT. SimilarMinRttThreshold = 1.125 // Congestion window gain for QUIC BBR during PROBE_BW phase. DefaultCongestionWindowGainConst = 2.0 )
var (
InfiniteBandwidth = Bandwidth(math.MaxUint64)
)
var (
InfiniteRTT = time.Duration(math.MaxInt64)
)
Functions ¶
func MaxFilter ¶
Compares two values and returns true if the first is greater than or equal to the second.
func MinFilter ¶
Compares two values and returns true if the first is less than or equal to the second.
func NewBBRSender ¶
func NewCubicSender ¶
func NewCubicSender(clock Clock, rttStats *RTTStats, reno bool, initialCongestionWindow, initialMaxCongestionWindow protocol.ByteCount) *cubicSender
NewCubicSender makes a new cubic sender
func SentPacketToSendTimeState ¶
func SentPacketToSendTimeState(sentPacket *ConnectionStateOnSentPacket, sendTimeState *SendTimeState)
SentPacketToSendTimeState Copy a subset of the (private) ConnectionStateOnSentPacket to the (public) SendTimeState. Always set send_time_state->is_valid to true.
Types ¶
type Bandwidth ¶
type Bandwidth uint64
Bandwidth of a connection
const ( // BitsPerSecond is 1 bit per second BitsPerSecond Bandwidth = 1 // BytesPerSecond is 1 byte per second BytesPerSecond = 8 * BitsPerSecond )
type BandwidthSample ¶
type BandwidthSample struct {
// contains filtered or unexported fields
}
BandwidthSample
func NewBandwidthSample ¶
func NewBandwidthSample() *BandwidthSample
type BandwidthSampler ¶
type BandwidthSampler struct {
// contains filtered or unexported fields
}
BandwidthSampler keeps track of sent and acknowledged packets and outputs a bandwidth sample for every packet acknowledged. The samples are taken for individual packets, and are not filtered; the consumer has to filter the bandwidth samples itself. In certain cases, the sampler will locally severely underestimate the bandwidth, hence a maximum filter with a size of at least one RTT is recommended.
This class bases its samples on the slope of two curves: the number of bytes sent over time, and the number of bytes acknowledged as received over time. It produces a sample of both slopes for every packet that gets acknowledged, based on a slope between two points on each of the corresponding curves. Note that due to the packet loss, the number of bytes on each curve might get further and further away from each other, meaning that it is not feasible to compare byte values coming from different curves with each other.
The obvious points for measuring slope sample are the ones corresponding to the packet that was just acknowledged. Let us denote them as S_1 (point at which the current packet was sent) and A_1 (point at which the current packet was acknowledged). However, taking a slope requires two points on each line, so estimating bandwidth requires picking a packet in the past with respect to which the slope is measured.
For that purpose, BandwidthSampler always keeps track of the most recently acknowledged packet, and records it together with every outgoing packet. When a packet gets acknowledged (A_1), it has not only information about when it itself was sent (S_1), but also the information about the latest acknowledged packet right before it was sent (S_0 and A_0).
Based on that data, send and ack rate are estimated as:
send_rate = (bytes(S_1) - bytes(S_0)) / (time(S_1) - time(S_0)) ack_rate = (bytes(A_1) - bytes(A_0)) / (time(A_1) - time(A_0))
Here, the ack rate is intuitively the rate we want to treat as bandwidth. However, in certain cases (e.g. ack compression) the ack rate at a point may end up higher than the rate at which the data was originally sent, which is not indicative of the real bandwidth. Hence, we use the send rate as an upper bound, and the sample value is
rate_sample = min(send_rate, ack_rate)
An important edge case handled by the sampler is tracking the app-limited samples. There are multiple meaning of "app-limited" used interchangeably, hence it is important to understand and to be able to distinguish between them.
Meaning 1: connection state. The connection is said to be app-limited when there is no outstanding data to send. This means that certain bandwidth samples in the future would not be an accurate indication of the link capacity, and it is important to inform consumer about that. Whenever connection becomes app-limited, the sampler is notified via OnAppLimited() method.
Meaning 2: a phase in the bandwidth sampler. As soon as the bandwidth sampler becomes notified about the connection being app-limited, it enters app-limited phase. In that phase, all *sent* packets are marked as app-limited. Note that the connection itself does not have to be app-limited during the app-limited phase, and in fact it will not be (otherwise how would it send packets?). The boolean flag below indicates whether the sampler is in that phase.
Meaning 3: a flag on the sent packet and on the sample. If a sent packet is sent during the app-limited phase, the resulting sample related to the packet will be marked as app-limited.
With the terminology issue out of the way, let us consider the question of what kind of situation it addresses.
Consider a scenario where we first send packets 1 to 20 at a regular bandwidth, and then immediately run out of data. After a few seconds, we send packets 21 to 60, and only receive ack for 21 between sending packets 40 and 41. In this case, when we sample bandwidth for packets 21 to 40, the S_0/A_0 we use to compute the slope is going to be packet 20, a few seconds apart from the current packet, hence the resulting estimate would be extremely low and not indicative of anything. Only at packet 41 the S_0/A_0 will become 21, meaning that the bandwidth sample would exclude the quiescence.
Based on the analysis of that scenario, we implement the following rule: once OnAppLimited() is called, all sent packets will produce app-limited samples up until an ack for a packet that was sent after OnAppLimited() was called. Note that while the scenario above is not the only scenario when the connection is app-limited, the approach works in other cases too.
func NewBandwidthSampler ¶
func NewBandwidthSampler() *BandwidthSampler
func (*BandwidthSampler) OnAppLimited ¶
func (s *BandwidthSampler) OnAppLimited()
OnAppLimited Informs the sampler that the connection is currently app-limited, causing the sampler to enter the app-limited phase. The phase will expire by itself.
func (*BandwidthSampler) OnPacketAcked ¶
func (s *BandwidthSampler) OnPacketAcked(ackTime time.Time, lastAckedPacket protocol.PacketNumber) *BandwidthSample
OnPacketAcked Notifies the sampler that the |lastAckedPacket| is acknowledged. Returns a bandwidth sample. If no bandwidth sample is available, QuicBandwidth::Zero() is returned.
func (*BandwidthSampler) OnPacketLost ¶
func (s *BandwidthSampler) OnPacketLost(packetNumber protocol.PacketNumber) SendTimeState
OnPacketLost Informs the sampler that a packet is considered lost and it should no longer keep track of it.
func (*BandwidthSampler) OnPacketSent ¶
func (s *BandwidthSampler) OnPacketSent(sentTime time.Time, lastSentPacket protocol.PacketNumber, sentBytes, bytesInFlight protocol.ByteCount, hasRetransmittableData bool)
OnPacketSent Inputs the sent packet information into the sampler. Assumes that all packets are sent in order. The information about the packet will not be released from the sampler until it the packet is either acknowledged or declared lost.
type CongestionEvent ¶
type ConnectionStateOnSentPacket ¶
type ConnectionStateOnSentPacket struct {
// contains filtered or unexported fields
}
ConnectionStateOnSentPacket represents the information about a sent packet and the state of the connection at the moment the packet was sent, specifically the information about the most recently acknowledged packet at that moment.
func NewConnectionStateOnSentPacket ¶
func NewConnectionStateOnSentPacket(packetNumber protocol.PacketNumber, sentTime time.Time, bytes protocol.ByteCount, sampler *BandwidthSampler) *ConnectionStateOnSentPacket
type ConnectionStates ¶
type ConnectionStates struct {
// contains filtered or unexported fields
}
ConnectionStates Record of the connection state at the point where each packet in flight was sent, indexed by the packet number. FIXME: using LinkedList replace map to fast remove all the packets lower than the specified packet number.
func (*ConnectionStates) Get ¶
func (s *ConnectionStates) Get(packetNumber protocol.PacketNumber) *ConnectionStateOnSentPacket
func (*ConnectionStates) Insert ¶
func (s *ConnectionStates) Insert(packetNumber protocol.PacketNumber, sentTime time.Time, bytes protocol.ByteCount, sampler *BandwidthSampler) bool
func (*ConnectionStates) Remove ¶
func (s *ConnectionStates) Remove(packetNumber protocol.PacketNumber) (bool, *ConnectionStateOnSentPacket)
type Cubic ¶
type Cubic struct {
// contains filtered or unexported fields
}
Cubic implements the cubic algorithm from TCP
func (*Cubic) CongestionWindowAfterAck ¶
func (c *Cubic) CongestionWindowAfterAck( ackedBytes protocol.ByteCount, currentCongestionWindow protocol.ByteCount, delayMin time.Duration, eventTime time.Time, ) protocol.ByteCount
CongestionWindowAfterAck computes a new congestion window to use after a received ACK. Returns the new congestion window in packets. The new congestion window follows a cubic function that depends on the time passed since last packet loss.
func (*Cubic) CongestionWindowAfterPacketLoss ¶
func (c *Cubic) CongestionWindowAfterPacketLoss(currentCongestionWindow protocol.ByteCount) protocol.ByteCount
CongestionWindowAfterPacketLoss computes a new congestion window to use after a loss event. Returns the new congestion window in packets. The new congestion window is a multiplicative decrease of our current window.
func (*Cubic) OnApplicationLimited ¶
func (c *Cubic) OnApplicationLimited()
OnApplicationLimited is called on ack arrival when sender is unable to use the available congestion window. Resets Cubic state during quiescence.
func (*Cubic) Reset ¶
func (c *Cubic) Reset()
Reset is called after a timeout to reset the cubic state
func (*Cubic) SetNumConnections ¶
SetNumConnections sets the number of emulated connections
type DefaultClock ¶
type DefaultClock struct{}
DefaultClock implements the Clock interface using the Go stdlib clock.
type HybridSlowStart ¶
type HybridSlowStart struct {
// contains filtered or unexported fields
}
HybridSlowStart implements the TCP hybrid slow start algorithm
func (*HybridSlowStart) IsEndOfRound ¶
func (s *HybridSlowStart) IsEndOfRound(ack protocol.PacketNumber) bool
IsEndOfRound returns true if this ack is the last packet number of our current slow start round.
func (*HybridSlowStart) OnPacketAcked ¶
func (s *HybridSlowStart) OnPacketAcked(ackedPacketNumber protocol.PacketNumber)
OnPacketAcked gets invoked after ShouldExitSlowStart, so it's best to end the round when the final packet of the burst is received and start it on the next incoming ack.
func (*HybridSlowStart) OnPacketSent ¶
func (s *HybridSlowStart) OnPacketSent(packetNumber protocol.PacketNumber)
OnPacketSent is called when a packet was sent
func (*HybridSlowStart) ShouldExitSlowStart ¶
func (s *HybridSlowStart) ShouldExitSlowStart(latestRTT time.Duration, minRTT time.Duration, congestionWindow protocol.ByteCount) bool
ShouldExitSlowStart should be called on every new ack frame, since a new RTT measurement can be made then. rtt: the RTT for this ack packet. minRTT: is the lowest delay (RTT) we have seen during the session. congestionWindow: the congestion window in packets.
func (*HybridSlowStart) StartReceiveRound ¶
func (s *HybridSlowStart) StartReceiveRound(lastSent protocol.PacketNumber)
StartReceiveRound is called for the start of each receive round (burst) in the slow start phase.
func (*HybridSlowStart) Started ¶
func (s *HybridSlowStart) Started() bool
Started returns true if started
type PrrSender ¶
type PrrSender struct {
// contains filtered or unexported fields
}
PrrSender implements the Proportional Rate Reduction (PRR) per RFC 6937
func (*PrrSender) CanSend ¶
func (p *PrrSender) CanSend(congestionWindow, bytesInFlight, slowstartThreshold protocol.ByteCount) bool
CanSend returns if packets can be sent
func (*PrrSender) OnPacketAcked ¶
OnPacketAcked should be called after a packet was acked
func (*PrrSender) OnPacketLost ¶
OnPacketLost should be called on the first loss that triggers a recovery period and all other methods in this class should only be called when in recovery.
func (*PrrSender) OnPacketSent ¶
OnPacketSent should be called after a packet was sent
type RTTStats ¶
type RTTStats struct {
// contains filtered or unexported fields
}
RTTStats provides round-trip statistics
func NewRTTStats ¶
func NewRTTStats() *RTTStats
NewRTTStats makes a properly initialized RTTStats object
func (*RTTStats) ExpireSmoothedMetrics ¶
func (r *RTTStats) ExpireSmoothedMetrics()
ExpireSmoothedMetrics causes the smoothed_rtt to be increased to the latest_rtt if the latest_rtt is larger. The mean deviation is increased to the most recent deviation if it's larger.
func (*RTTStats) LatestRTT ¶
LatestRTT returns the most recent rtt measurement. May return Zero if no valid updates have occurred.
func (*RTTStats) MeanDeviation ¶
MeanDeviation gets the mean deviation
func (*RTTStats) MinRTT ¶
MinRTT Returns the minRTT for the entire connection. May return Zero if no valid updates have occurred.
func (*RTTStats) OnConnectionMigration ¶
func (r *RTTStats) OnConnectionMigration()
OnConnectionMigration is called when connection migrates and rtt measurement needs to be reset.
func (*RTTStats) SmoothedOrInitialRTT ¶
SmoothedOrInitialRTT returns the EWMA smoothed RTT for the connection. If no valid updates have occurred, it returns the initial RTT.
func (*RTTStats) SmoothedRTT ¶
SmoothedRTT returns the EWMA smoothed RTT for the connection. May return Zero if no valid updates have occurred.
type SendAlgorithm ¶
type SendAlgorithm interface { TimeUntilSend(bytesInFlight protocol.ByteCount) time.Duration OnPacketSent(sentTime time.Time, bytesInFlight protocol.ByteCount, packetNumber protocol.PacketNumber, bytes protocol.ByteCount, isRetransmittable bool) CanSend(bytesInFlight protocol.ByteCount) bool MaybeExitSlowStart() OnPacketAcked(number protocol.PacketNumber, ackedBytes protocol.ByteCount, priorInFlight protocol.ByteCount, eventTime time.Time) OnPacketLost(number protocol.PacketNumber, lostBytes protocol.ByteCount, priorInFlight protocol.ByteCount) OnRetransmissionTimeout(packetsRetransmitted bool) }
A SendAlgorithm performs congestion control
type SendAlgorithmWithDebugInfos ¶
type SendAlgorithmWithDebugInfos interface { SendAlgorithm GetCongestionWindow() protocol.ByteCount }
A SendAlgorithmWithDebugInfos is a SendAlgorithm that exposes some debug infos
type SendTimeState ¶
type SendTimeState struct {
// contains filtered or unexported fields
}
SendTimeState is a subset of ConnectionStateOnSentPacket which is returned to the caller when the packet is acked or lost.
type WindowedFilter ¶
type WindowedFilter struct {
// contains filtered or unexported fields
}
WindowedFilter Use the following to construct a windowed filter object of type T. For example, a min filter using QuicTime as the time type:
WindowedFilter<T, MinFilter<T>, QuicTime, QuicTime::Delta> ObjectName;
A max filter using 64-bit integers as the time type:
WindowedFilter<T, MaxFilter<T>, uint64_t, int64_t> ObjectName;
Specifically, this template takes four arguments:
- T -- type of the measurement that is being filtered.
- Compare -- MinFilter<T> or MaxFilter<T>, depending on the type of filter desired.
- TimeT -- the type used to represent timestamps.
- TimeDeltaT -- the type used to represent continuous time intervals between two timestamps. Has to be the type of (a - b) if both |a| and |b| are of type TimeT.
func NewWindowedFilter ¶
func NewWindowedFilter(windowLength int64, comparator func(int64, int64) bool) *WindowedFilter
func (*WindowedFilter) GetBest ¶
func (f *WindowedFilter) GetBest() int64
func (*WindowedFilter) GetSecondBest ¶
func (f *WindowedFilter) GetSecondBest() int64
func (*WindowedFilter) GetThirdBest ¶
func (f *WindowedFilter) GetThirdBest() int64
func (*WindowedFilter) Reset ¶
func (f *WindowedFilter) Reset(newSample int64, newTime int64)
func (*WindowedFilter) SetWindowLength ¶
func (f *WindowedFilter) SetWindowLength(windowLength int64)
Changes the window length. Does not update any current samples.
func (*WindowedFilter) Update ¶
func (f *WindowedFilter) Update(sample int64, time int64)