mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
The RTTStats are used by the logging package. In order to instrument the congestion package, the RTTStats can't be part of that package any more (to avoid an import loop).
127 lines
3.9 KiB
Go
127 lines
3.9 KiB
Go
package utils
|
|
|
|
import (
|
|
"time"
|
|
|
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
|
)
|
|
|
|
const (
|
|
rttAlpha = 0.125
|
|
oneMinusAlpha = 1 - rttAlpha
|
|
rttBeta = 0.25
|
|
oneMinusBeta = 1 - rttBeta
|
|
// The default RTT used before an RTT sample is taken.
|
|
defaultInitialRTT = 100 * time.Millisecond
|
|
)
|
|
|
|
// RTTStats provides round-trip statistics
|
|
type RTTStats struct {
|
|
hasMeasurement bool
|
|
|
|
minRTT time.Duration
|
|
latestRTT time.Duration
|
|
smoothedRTT time.Duration
|
|
meanDeviation time.Duration
|
|
|
|
maxAckDelay time.Duration
|
|
}
|
|
|
|
// NewRTTStats makes a properly initialized RTTStats object
|
|
func NewRTTStats() *RTTStats {
|
|
return &RTTStats{}
|
|
}
|
|
|
|
// MinRTT Returns the minRTT for the entire connection.
|
|
// May return Zero if no valid updates have occurred.
|
|
func (r *RTTStats) MinRTT() time.Duration { return r.minRTT }
|
|
|
|
// LatestRTT returns the most recent rtt measurement.
|
|
// May return Zero if no valid updates have occurred.
|
|
func (r *RTTStats) LatestRTT() time.Duration { return r.latestRTT }
|
|
|
|
// SmoothedRTT returns the smoothed RTT for the connection.
|
|
// May return Zero if no valid updates have occurred.
|
|
func (r *RTTStats) SmoothedRTT() time.Duration { return r.smoothedRTT }
|
|
|
|
// MeanDeviation gets the mean deviation
|
|
func (r *RTTStats) MeanDeviation() time.Duration { return r.meanDeviation }
|
|
|
|
// MaxAckDelay gets the max_ack_delay advertised by the peer
|
|
func (r *RTTStats) MaxAckDelay() time.Duration { return r.maxAckDelay }
|
|
|
|
// PTO gets the probe timeout duration.
|
|
func (r *RTTStats) PTO(includeMaxAckDelay bool) time.Duration {
|
|
if r.SmoothedRTT() == 0 {
|
|
return 2 * defaultInitialRTT
|
|
}
|
|
pto := r.SmoothedRTT() + MaxDuration(4*r.MeanDeviation(), protocol.TimerGranularity)
|
|
if includeMaxAckDelay {
|
|
pto += r.MaxAckDelay()
|
|
}
|
|
return pto
|
|
}
|
|
|
|
// UpdateRTT updates the RTT based on a new sample.
|
|
func (r *RTTStats) UpdateRTT(sendDelta, ackDelay time.Duration, now time.Time) {
|
|
if sendDelta == InfDuration || sendDelta <= 0 {
|
|
return
|
|
}
|
|
|
|
// Update r.minRTT first. r.minRTT does not use an rttSample corrected for
|
|
// ackDelay but the raw observed sendDelta, since poor clock granularity at
|
|
// the client may cause a high ackDelay to result in underestimation of the
|
|
// r.minRTT.
|
|
if r.minRTT == 0 || r.minRTT > sendDelta {
|
|
r.minRTT = sendDelta
|
|
}
|
|
|
|
// Correct for ackDelay if information received from the peer results in a
|
|
// an RTT sample at least as large as minRTT. Otherwise, only use the
|
|
// sendDelta.
|
|
sample := sendDelta
|
|
if sample-r.minRTT >= ackDelay {
|
|
sample -= ackDelay
|
|
}
|
|
r.latestRTT = sample
|
|
// First time call.
|
|
if !r.hasMeasurement {
|
|
r.hasMeasurement = true
|
|
r.smoothedRTT = sample
|
|
r.meanDeviation = sample / 2
|
|
} else {
|
|
r.meanDeviation = time.Duration(oneMinusBeta*float32(r.meanDeviation/time.Microsecond)+rttBeta*float32(AbsDuration(r.smoothedRTT-sample)/time.Microsecond)) * time.Microsecond
|
|
r.smoothedRTT = time.Duration((float32(r.smoothedRTT/time.Microsecond)*oneMinusAlpha)+(float32(sample/time.Microsecond)*rttAlpha)) * time.Microsecond
|
|
}
|
|
}
|
|
|
|
// SetMaxAckDelay sets the max_ack_delay
|
|
func (r *RTTStats) SetMaxAckDelay(mad time.Duration) {
|
|
r.maxAckDelay = mad
|
|
}
|
|
|
|
// SetInitialRTT sets the initial RTT.
|
|
// It is used during the 0-RTT handshake when restoring the RTT stats from the session state.
|
|
func (r *RTTStats) SetInitialRTT(t time.Duration) {
|
|
if r.hasMeasurement {
|
|
panic("initial RTT set after first measurement")
|
|
}
|
|
r.smoothedRTT = t
|
|
r.latestRTT = t
|
|
}
|
|
|
|
// OnConnectionMigration is called when connection migrates and rtt measurement needs to be reset.
|
|
func (r *RTTStats) OnConnectionMigration() {
|
|
r.latestRTT = 0
|
|
r.minRTT = 0
|
|
r.smoothedRTT = 0
|
|
r.meanDeviation = 0
|
|
}
|
|
|
|
// 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 (r *RTTStats) ExpireSmoothedMetrics() {
|
|
r.meanDeviation = MaxDuration(r.meanDeviation, AbsDuration(r.smoothedRTT-r.latestRTT))
|
|
r.smoothedRTT = MaxDuration(r.smoothedRTT, r.latestRTT)
|
|
}
|