2017-05-27 13:30:11 -06:00
|
|
|
package congestion
|
|
|
|
|
|
|
|
import (
|
2018-02-17 13:29:53 +08:00
|
|
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
2017-05-27 13:30:11 -06:00
|
|
|
)
|
|
|
|
|
|
|
|
// PrrSender implements the Proportional Rate Reduction (PRR) per RFC 6937
|
|
|
|
type PrrSender struct {
|
|
|
|
bytesSentSinceLoss protocol.ByteCount
|
|
|
|
bytesDeliveredSinceLoss protocol.ByteCount
|
|
|
|
ackCountSinceLoss protocol.ByteCount
|
|
|
|
bytesInFlightBeforeLoss protocol.ByteCount
|
|
|
|
}
|
|
|
|
|
|
|
|
// OnPacketSent should be called after a packet was sent
|
|
|
|
func (p *PrrSender) OnPacketSent(sentBytes protocol.ByteCount) {
|
|
|
|
p.bytesSentSinceLoss += sentBytes
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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.
|
2018-09-03 04:18:54 +07:00
|
|
|
func (p *PrrSender) OnPacketLost(priorInFlight protocol.ByteCount) {
|
2017-05-27 13:30:11 -06:00
|
|
|
p.bytesSentSinceLoss = 0
|
2018-09-03 04:18:54 +07:00
|
|
|
p.bytesInFlightBeforeLoss = priorInFlight
|
2017-05-27 13:30:11 -06:00
|
|
|
p.bytesDeliveredSinceLoss = 0
|
|
|
|
p.ackCountSinceLoss = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
// OnPacketAcked should be called after a packet was acked
|
|
|
|
func (p *PrrSender) OnPacketAcked(ackedBytes protocol.ByteCount) {
|
|
|
|
p.bytesDeliveredSinceLoss += ackedBytes
|
|
|
|
p.ackCountSinceLoss++
|
|
|
|
}
|
|
|
|
|
2018-09-03 04:18:54 +07:00
|
|
|
// CanSend returns if packets can be sent
|
|
|
|
func (p *PrrSender) CanSend(congestionWindow, bytesInFlight, slowstartThreshold protocol.ByteCount) bool {
|
2017-05-27 13:30:11 -06:00
|
|
|
// Return QuicTime::Zero In order to ensure limited transmit always works.
|
|
|
|
if p.bytesSentSinceLoss == 0 || bytesInFlight < protocol.DefaultTCPMSS {
|
2018-09-03 04:18:54 +07:00
|
|
|
return true
|
2017-05-27 13:30:11 -06:00
|
|
|
}
|
|
|
|
if congestionWindow > bytesInFlight {
|
|
|
|
// During PRR-SSRB, limit outgoing packets to 1 extra MSS per ack, instead
|
|
|
|
// of sending the entire available window. This prevents burst retransmits
|
|
|
|
// when more packets are lost than the CWND reduction.
|
|
|
|
// limit = MAX(prr_delivered - prr_out, DeliveredData) + MSS
|
2018-09-03 04:18:54 +07:00
|
|
|
return p.bytesDeliveredSinceLoss+p.ackCountSinceLoss*protocol.DefaultTCPMSS > p.bytesSentSinceLoss
|
2017-05-27 13:30:11 -06:00
|
|
|
}
|
|
|
|
// Implement Proportional Rate Reduction (RFC6937).
|
|
|
|
// Checks a simplified version of the PRR formula that doesn't use division:
|
|
|
|
// AvailableSendWindow =
|
|
|
|
// CEIL(prr_delivered * ssthresh / BytesInFlightAtLoss) - prr_sent
|
2018-09-03 04:18:54 +07:00
|
|
|
return p.bytesDeliveredSinceLoss*slowstartThreshold > p.bytesSentSinceLoss*p.bytesInFlightBeforeLoss
|
2017-05-27 13:30:11 -06:00
|
|
|
}
|