package ackhandler import ( "time" "github.com/lucas-clemente/quic-go/internal/protocol" "github.com/lucas-clemente/quic-go/internal/wire" ) type receivedPacketHandler struct { largestObserved protocol.PacketNumber ignoreBelow protocol.PacketNumber largestObservedReceivedTime time.Time packetHistory *receivedPacketHistory ackSendDelay time.Duration packetsReceivedSinceLastAck int retransmittablePacketsReceivedSinceLastAck int ackQueued bool ackAlarm time.Time lastAck *wire.AckFrame version protocol.VersionNumber } // NewReceivedPacketHandler creates a new receivedPacketHandler func NewReceivedPacketHandler(version protocol.VersionNumber) ReceivedPacketHandler { return &receivedPacketHandler{ packetHistory: newReceivedPacketHistory(), ackSendDelay: protocol.AckSendDelay, version: version, } } func (h *receivedPacketHandler) ReceivedPacket(packetNumber protocol.PacketNumber, rcvTime time.Time, shouldInstigateAck bool) error { if packetNumber > h.largestObserved { h.largestObserved = packetNumber h.largestObservedReceivedTime = rcvTime } if packetNumber < h.ignoreBelow { return nil } if err := h.packetHistory.ReceivedPacket(packetNumber); err != nil { return err } h.maybeQueueAck(packetNumber, rcvTime, shouldInstigateAck) return nil } // IgnoreBelow sets a lower limit for acking packets. // Packets with packet numbers smaller than p will not be acked. func (h *receivedPacketHandler) IgnoreBelow(p protocol.PacketNumber) { h.ignoreBelow = p h.packetHistory.DeleteBelow(p) } func (h *receivedPacketHandler) maybeQueueAck(packetNumber protocol.PacketNumber, rcvTime time.Time, shouldInstigateAck bool) { h.packetsReceivedSinceLastAck++ if shouldInstigateAck { h.retransmittablePacketsReceivedSinceLastAck++ } // always ack the first packet if h.lastAck == nil { h.ackQueued = true } // if the packet number is smaller than the largest acked packet, it must have been reported missing with the last ACK // note that it cannot be a duplicate because they're already filtered out by ReceivedPacket() if h.lastAck != nil && packetNumber < h.lastAck.LargestAcked { h.ackQueued = true } // check if a new missing range above the previously was created if h.lastAck != nil && h.packetHistory.GetHighestAckRange().First > h.lastAck.LargestAcked { h.ackQueued = true } if !h.ackQueued && shouldInstigateAck { if h.retransmittablePacketsReceivedSinceLastAck >= protocol.RetransmittablePacketsBeforeAck { h.ackQueued = true } else { if h.ackAlarm.IsZero() { h.ackAlarm = rcvTime.Add(h.ackSendDelay) } } } if h.ackQueued { // cancel the ack alarm h.ackAlarm = time.Time{} } } func (h *receivedPacketHandler) GetAckFrame() *wire.AckFrame { if !h.ackQueued && (h.ackAlarm.IsZero() || h.ackAlarm.After(time.Now())) { return nil } ackRanges := h.packetHistory.GetAckRanges() ack := &wire.AckFrame{ LargestAcked: h.largestObserved, LowestAcked: ackRanges[len(ackRanges)-1].First, PacketReceivedTime: h.largestObservedReceivedTime, } if len(ackRanges) > 1 { ack.AckRanges = ackRanges } h.lastAck = ack h.ackAlarm = time.Time{} h.ackQueued = false h.packetsReceivedSinceLastAck = 0 h.retransmittablePacketsReceivedSinceLastAck = 0 return ack } func (h *receivedPacketHandler) GetAlarmTimeout() time.Time { return h.ackAlarm }