2023-05-10 15:50:04 +02:00
|
|
|
# Copyright (c) 2021-2023 Status Research & Development GmbH
|
2021-12-10 11:12:24 +01:00
|
|
|
# Licensed and distributed under either of
|
|
|
|
# * MIT license (license terms in the root directory or at https://opensource.org/licenses/MIT).
|
|
|
|
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0).
|
|
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
|
2023-05-10 15:50:04 +02:00
|
|
|
{.push raises: [].}
|
2021-12-10 11:12:24 +01:00
|
|
|
|
2021-12-09 10:52:21 +01:00
|
|
|
import
|
|
|
|
chronos,
|
|
|
|
./utp_utils
|
|
|
|
|
|
|
|
const targetDelay = milliseconds(100)
|
|
|
|
|
|
|
|
# explanation from reference impl:
|
|
|
|
# number of bytes to increase max window size by, per RTT. This is
|
|
|
|
# scaled down linearly proportional to off_target. i.e. if all packets
|
|
|
|
# in one window have 0 delay, window size will increase by this number.
|
|
|
|
# Typically it's less. TCP increases one MSS per RTT, which is 1500
|
|
|
|
const maxCwndIncreaseBytesPerRtt = 3000
|
|
|
|
|
2022-01-27 11:07:40 +01:00
|
|
|
const minWindowSize* = 10
|
2021-12-09 10:52:21 +01:00
|
|
|
|
|
|
|
proc applyCongestionControl*(
|
|
|
|
currentMaxWindowSize: uint32,
|
|
|
|
currentSlowStart: bool,
|
2022-11-16 10:44:00 -06:00
|
|
|
currentSlowStartThreshold: uint32,
|
2021-12-09 10:52:21 +01:00
|
|
|
maxSndBufferSize: uint32,
|
|
|
|
currentPacketSize: uint32,
|
|
|
|
actualDelay: Duration,
|
|
|
|
numOfAckedBytes: uint32,
|
|
|
|
minRtt: Duration,
|
|
|
|
calculatedDelay: Duration,
|
|
|
|
clockDrift: int32
|
2021-12-20 13:14:50 +01:00
|
|
|
): (uint32, uint32, bool) =
|
2021-12-09 10:52:21 +01:00
|
|
|
if (actualDelay.isZero() or minRtt.isZero() or numOfAckedBytes == 0):
|
2022-11-16 10:44:00 -06:00
|
|
|
return (currentMaxWindowSize, currentSlowStartThreshold, currentSlowStart)
|
2021-12-20 13:14:50 +01:00
|
|
|
|
2021-12-09 10:52:21 +01:00
|
|
|
let ourDelay = min(minRtt, calculatedDelay)
|
|
|
|
|
|
|
|
let target = targetDelay
|
|
|
|
|
|
|
|
# Rationale from C reference impl:
|
|
|
|
# this is here to compensate for very large clock drift that affects
|
|
|
|
# the congestion controller into giving certain endpoints an unfair
|
|
|
|
# share of the bandwidth. We have an estimate of the clock drift
|
|
|
|
# (clock_drift). The unit of this is microseconds per 5 seconds.
|
|
|
|
# empirically, a reasonable cut-off appears to be about 200000
|
|
|
|
# (which is pretty high). The main purpose is to compensate for
|
|
|
|
# people trying to "cheat" uTP by making their clock run slower,
|
|
|
|
# and this definitely catches that without any risk of false positives
|
|
|
|
# if clock_drift < -200000 start applying a penalty delay proportional
|
2022-11-16 10:44:00 -06:00
|
|
|
# to how far beyond -200000 the clock drift is
|
2021-12-09 10:52:21 +01:00
|
|
|
let clockDriftPenalty: int64 =
|
|
|
|
if (clockDrift < -200000):
|
|
|
|
let penalty = (-clockDrift - 200000) div 7
|
|
|
|
penalty
|
|
|
|
else:
|
|
|
|
0
|
|
|
|
|
|
|
|
let offTarget = target.microseconds() - (ourDelay.microseconds() + clockDriftPenalty)
|
|
|
|
|
|
|
|
# calculations from reference impl:
|
|
|
|
# double window_factor = (double)min(bytes_acked, max_window) / (double)max(max_window, bytes_acked);
|
|
|
|
# double delay_factor = off_target / target;
|
|
|
|
# double scaled_gain = MAX_CWND_INCREASE_BYTES_PER_RTT * window_factor * delay_factor;
|
2021-12-20 13:14:50 +01:00
|
|
|
|
2021-12-09 10:52:21 +01:00
|
|
|
let windowFactor = float64(min(numOfAckedBytes, currentMaxWindowSize)) / float64(max(currentMaxWindowSize, numOfAckedBytes))
|
|
|
|
|
|
|
|
let delayFactor = float64(offTarget) / float64(target.microseconds())
|
|
|
|
|
|
|
|
let scaledGain = maxCwndIncreaseBytesPerRtt * windowFactor * delayFactor
|
|
|
|
|
|
|
|
let scaledWindow = float64(currentMaxWindowSize) + scaledGain
|
|
|
|
|
2021-12-20 13:14:50 +01:00
|
|
|
let ledbatCwnd: uint32 =
|
2021-12-09 10:52:21 +01:00
|
|
|
if scaledWindow < minWindowSize:
|
|
|
|
uint32(minWindowSize)
|
|
|
|
else:
|
|
|
|
uint32(scaledWindow)
|
|
|
|
|
|
|
|
var newSlowStart = currentSlowStart
|
|
|
|
var newMaxWindowSize = currentMaxWindowSize
|
2022-11-16 10:44:00 -06:00
|
|
|
var newSlowStartThreshold = currentSlowStartThreshold
|
2021-12-09 10:52:21 +01:00
|
|
|
|
|
|
|
if currentSlowStart:
|
|
|
|
let slowStartCwnd = currentMaxWindowSize + uint32(windowFactor * float64(currentPacketSize))
|
|
|
|
|
2022-11-16 10:44:00 -06:00
|
|
|
if (slowStartCwnd > currentSlowStartThreshold):
|
2021-12-09 10:52:21 +01:00
|
|
|
newSlowStart = false
|
|
|
|
elif float64(ourDelay.microseconds()) > float64(target.microseconds()) * 0.9:
|
2022-11-16 10:44:00 -06:00
|
|
|
# we are just a little under target delay, discontinue slows start
|
2021-12-09 10:52:21 +01:00
|
|
|
newSlowStart = false
|
2022-11-16 10:44:00 -06:00
|
|
|
newSlowStartThreshold = currentMaxWindowSize
|
2021-12-09 10:52:21 +01:00
|
|
|
else:
|
|
|
|
newMaxWindowSize = max(slowStartCwnd, ledbatCwnd)
|
|
|
|
else:
|
|
|
|
newMaxWindowSize = ledbatCwnd
|
|
|
|
|
|
|
|
newMaxWindowSize = clamp(newMaxWindowSize, minWindowSize, maxSndBufferSize)
|
|
|
|
|
2022-11-16 10:44:00 -06:00
|
|
|
(newMaxWindowSize, newSlowStartThreshold, newSlowStart)
|