mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
ackhandler: refactor loss detection timer reset logic (#4893)
This commit is contained in:
parent
5c545cac23
commit
bf3ccd1fb3
1 changed files with 47 additions and 49 deletions
|
@ -53,6 +53,12 @@ func newPacketNumberSpace(initialPN protocol.PacketNumber, isAppData bool) *pack
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type alarmTimer struct {
|
||||||
|
Time time.Time
|
||||||
|
TimerType logging.TimerType
|
||||||
|
EncryptionLevel protocol.EncryptionLevel
|
||||||
|
}
|
||||||
|
|
||||||
type sentPacketHandler struct {
|
type sentPacketHandler struct {
|
||||||
initialPackets *packetNumberSpace
|
initialPackets *packetNumberSpace
|
||||||
handshakePackets *packetNumberSpace
|
handshakePackets *packetNumberSpace
|
||||||
|
@ -90,7 +96,7 @@ type sentPacketHandler struct {
|
||||||
numProbesToSend int
|
numProbesToSend int
|
||||||
|
|
||||||
// The alarm timeout
|
// The alarm timeout
|
||||||
alarm time.Time
|
alarm alarmTimer
|
||||||
|
|
||||||
enableECN bool
|
enableECN bool
|
||||||
ecnTracker ecnHandler
|
ecnTracker ecnHandler
|
||||||
|
@ -548,61 +554,53 @@ func (h *sentPacketHandler) hasOutstandingCryptoPackets() bool {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHandler) hasOutstandingPackets() bool {
|
|
||||||
return h.appDataPackets.history.HasOutstandingPackets() || h.hasOutstandingCryptoPackets()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (h *sentPacketHandler) setLossDetectionTimer(now time.Time) {
|
func (h *sentPacketHandler) setLossDetectionTimer(now time.Time) {
|
||||||
oldAlarm := h.alarm // only needed in case tracing is enabled
|
oldAlarm := h.alarm // only needed in case tracing is enabled
|
||||||
|
newAlarm := h.lossDetectionTime(now)
|
||||||
|
h.alarm = newAlarm
|
||||||
|
|
||||||
|
if newAlarm.Time.IsZero() && !oldAlarm.Time.IsZero() {
|
||||||
|
h.logger.Debugf("Canceling loss detection timer.")
|
||||||
|
if h.tracer != nil && h.tracer.LossTimerCanceled != nil {
|
||||||
|
h.tracer.LossTimerCanceled()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if h.tracer != nil && h.tracer.SetLossTimer != nil && newAlarm != oldAlarm {
|
||||||
|
h.tracer.SetLossTimer(newAlarm.TimerType, newAlarm.EncryptionLevel, newAlarm.Time)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *sentPacketHandler) lossDetectionTime(now time.Time) alarmTimer {
|
||||||
|
// cancel the alarm if no packets are outstanding
|
||||||
|
if h.peerCompletedAddressValidation &&
|
||||||
|
!h.hasOutstandingCryptoPackets() && !h.appDataPackets.history.HasOutstandingPackets() {
|
||||||
|
return alarmTimer{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// cancel the alarm if amplification limited
|
||||||
|
if h.isAmplificationLimited() {
|
||||||
|
return alarmTimer{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// early retransmit timer or time loss detection
|
||||||
lossTime, encLevel := h.getLossTimeAndSpace()
|
lossTime, encLevel := h.getLossTimeAndSpace()
|
||||||
if !lossTime.IsZero() {
|
if !lossTime.IsZero() {
|
||||||
// Early retransmit timer or time loss detection.
|
return alarmTimer{
|
||||||
h.alarm = lossTime
|
Time: lossTime,
|
||||||
if h.tracer != nil && h.tracer.SetLossTimer != nil && h.alarm != oldAlarm {
|
TimerType: logging.TimerTypeACK,
|
||||||
h.tracer.SetLossTimer(logging.TimerTypeACK, encLevel, h.alarm)
|
EncryptionLevel: encLevel,
|
||||||
}
|
}
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cancel the alarm if amplification limited.
|
|
||||||
if h.isAmplificationLimited() {
|
|
||||||
h.alarm = time.Time{}
|
|
||||||
if !oldAlarm.IsZero() {
|
|
||||||
h.logger.Debugf("Canceling loss detection timer. Amplification limited.")
|
|
||||||
if h.tracer != nil && h.tracer.LossTimerCanceled != nil {
|
|
||||||
h.tracer.LossTimerCanceled()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Cancel the alarm if no packets are outstanding
|
|
||||||
if !h.hasOutstandingPackets() && h.peerCompletedAddressValidation {
|
|
||||||
h.alarm = time.Time{}
|
|
||||||
if !oldAlarm.IsZero() {
|
|
||||||
h.logger.Debugf("Canceling loss detection timer. No packets in flight.")
|
|
||||||
if h.tracer != nil && h.tracer.LossTimerCanceled != nil {
|
|
||||||
h.tracer.LossTimerCanceled()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// PTO alarm
|
|
||||||
ptoTime, encLevel, ok := h.getPTOTimeAndSpace(now)
|
ptoTime, encLevel, ok := h.getPTOTimeAndSpace(now)
|
||||||
if !ok {
|
if !ok {
|
||||||
if !oldAlarm.IsZero() {
|
return alarmTimer{}
|
||||||
h.alarm = time.Time{}
|
|
||||||
h.logger.Debugf("Canceling loss detection timer. No PTO needed..")
|
|
||||||
if h.tracer != nil && h.tracer.LossTimerCanceled != nil {
|
|
||||||
h.tracer.LossTimerCanceled()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
h.alarm = ptoTime
|
return alarmTimer{
|
||||||
if h.tracer != nil && h.tracer.SetLossTimer != nil && h.alarm != oldAlarm {
|
Time: ptoTime,
|
||||||
h.tracer.SetLossTimer(logging.TimerTypePTO, encLevel, h.alarm)
|
TimerType: logging.TimerTypePTO,
|
||||||
|
EncryptionLevel: encLevel,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -742,7 +740,7 @@ func (h *sentPacketHandler) OnLossDetectionTimeout(now time.Time) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHandler) GetLossDetectionTimeout() time.Time {
|
func (h *sentPacketHandler) GetLossDetectionTimeout() time.Time {
|
||||||
return h.alarm
|
return h.alarm.Time
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHandler) ECNMode(isShortHeaderPacket bool) protocol.ECN {
|
func (h *sentPacketHandler) ECNMode(isShortHeaderPacket bool) protocol.ECN {
|
||||||
|
@ -904,12 +902,12 @@ func (h *sentPacketHandler) ResetForRetry(now time.Time) {
|
||||||
h.initialPackets = newPacketNumberSpace(h.initialPackets.pns.Peek(), false)
|
h.initialPackets = newPacketNumberSpace(h.initialPackets.pns.Peek(), false)
|
||||||
h.appDataPackets = newPacketNumberSpace(h.appDataPackets.pns.Peek(), true)
|
h.appDataPackets = newPacketNumberSpace(h.appDataPackets.pns.Peek(), true)
|
||||||
oldAlarm := h.alarm
|
oldAlarm := h.alarm
|
||||||
h.alarm = time.Time{}
|
h.alarm = alarmTimer{}
|
||||||
if h.tracer != nil {
|
if h.tracer != nil {
|
||||||
if h.tracer.UpdatedPTOCount != nil {
|
if h.tracer.UpdatedPTOCount != nil {
|
||||||
h.tracer.UpdatedPTOCount(0)
|
h.tracer.UpdatedPTOCount(0)
|
||||||
}
|
}
|
||||||
if !oldAlarm.IsZero() && h.tracer.LossTimerCanceled != nil {
|
if !oldAlarm.Time.IsZero() && h.tracer.LossTimerCanceled != nil {
|
||||||
h.tracer.LossTimerCanceled()
|
h.tracer.LossTimerCanceled()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue