mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 12:47:36 +03:00
163 lines
4.3 KiB
Go
163 lines
4.3 KiB
Go
package ackhandler
|
|
|
|
import (
|
|
"fmt"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/quic-go/quic-go/internal/protocol"
|
|
"github.com/quic-go/quic-go/internal/utils"
|
|
list "github.com/quic-go/quic-go/internal/utils/linkedlist"
|
|
)
|
|
|
|
type sentPacketHistory struct {
|
|
rttStats *utils.RTTStats
|
|
outstandingPacketList *list.List[*Packet]
|
|
etcPacketList *list.List[*Packet]
|
|
packetMap map[protocol.PacketNumber]*list.Element[*Packet]
|
|
highestSent protocol.PacketNumber
|
|
}
|
|
|
|
var packetElementPool sync.Pool
|
|
|
|
func init() {
|
|
packetElementPool = *list.NewPool[*Packet]()
|
|
}
|
|
|
|
func newSentPacketHistory(rttStats *utils.RTTStats) *sentPacketHistory {
|
|
return &sentPacketHistory{
|
|
rttStats: rttStats,
|
|
outstandingPacketList: list.NewWithPool[*Packet](&packetElementPool),
|
|
etcPacketList: list.NewWithPool[*Packet](&packetElementPool),
|
|
packetMap: make(map[protocol.PacketNumber]*list.Element[*Packet]),
|
|
highestSent: protocol.InvalidPacketNumber,
|
|
}
|
|
}
|
|
|
|
func (h *sentPacketHistory) SentNonAckElicitingPacket(pn protocol.PacketNumber, encLevel protocol.EncryptionLevel, t time.Time) {
|
|
h.registerSentPacket(pn, encLevel, t)
|
|
}
|
|
|
|
func (h *sentPacketHistory) SentAckElicitingPacket(p *Packet) {
|
|
h.registerSentPacket(p.PacketNumber, p.EncryptionLevel, p.SendTime)
|
|
|
|
var el *list.Element[*Packet]
|
|
if p.outstanding() {
|
|
el = h.outstandingPacketList.PushBack(p)
|
|
} else {
|
|
el = h.etcPacketList.PushBack(p)
|
|
}
|
|
h.packetMap[p.PacketNumber] = el
|
|
}
|
|
|
|
func (h *sentPacketHistory) registerSentPacket(pn protocol.PacketNumber, encLevel protocol.EncryptionLevel, t time.Time) {
|
|
if pn <= h.highestSent {
|
|
panic("non-sequential packet number use")
|
|
}
|
|
// Skipped packet numbers.
|
|
for p := h.highestSent + 1; p < pn; p++ {
|
|
el := h.etcPacketList.PushBack(&Packet{
|
|
PacketNumber: p,
|
|
EncryptionLevel: encLevel,
|
|
SendTime: t,
|
|
skippedPacket: true,
|
|
})
|
|
h.packetMap[p] = el
|
|
}
|
|
h.highestSent = pn
|
|
}
|
|
|
|
// Iterate iterates through all packets.
|
|
func (h *sentPacketHistory) Iterate(cb func(*Packet) (cont bool, err error)) error {
|
|
cont := true
|
|
outstandingEl := h.outstandingPacketList.Front()
|
|
etcEl := h.etcPacketList.Front()
|
|
var el *list.Element[*Packet]
|
|
// whichever has the next packet number is returned first
|
|
for cont {
|
|
if outstandingEl == nil || (etcEl != nil && etcEl.Value.PacketNumber < outstandingEl.Value.PacketNumber) {
|
|
el = etcEl
|
|
} else {
|
|
el = outstandingEl
|
|
}
|
|
if el == nil {
|
|
return nil
|
|
}
|
|
if el == outstandingEl {
|
|
outstandingEl = outstandingEl.Next()
|
|
} else {
|
|
etcEl = etcEl.Next()
|
|
}
|
|
var err error
|
|
cont, err = cb(el.Value)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// FirstOutstanding returns the first outstanding packet.
|
|
func (h *sentPacketHistory) FirstOutstanding() *Packet {
|
|
el := h.outstandingPacketList.Front()
|
|
if el == nil {
|
|
return nil
|
|
}
|
|
return el.Value
|
|
}
|
|
|
|
func (h *sentPacketHistory) Len() int {
|
|
return len(h.packetMap)
|
|
}
|
|
|
|
func (h *sentPacketHistory) Remove(p protocol.PacketNumber) error {
|
|
el, ok := h.packetMap[p]
|
|
if !ok {
|
|
return fmt.Errorf("packet %d not found in sent packet history", p)
|
|
}
|
|
el.List().Remove(el)
|
|
delete(h.packetMap, p)
|
|
return nil
|
|
}
|
|
|
|
func (h *sentPacketHistory) HasOutstandingPackets() bool {
|
|
return h.outstandingPacketList.Len() > 0
|
|
}
|
|
|
|
func (h *sentPacketHistory) DeleteOldPackets(now time.Time) {
|
|
maxAge := 3 * h.rttStats.PTO(false)
|
|
var nextEl *list.Element[*Packet]
|
|
// we don't iterate outstandingPacketList, as we should not delete outstanding packets.
|
|
// being outstanding for more than 3*PTO should only happen in the case of drastic RTT changes.
|
|
for el := h.etcPacketList.Front(); el != nil; el = nextEl {
|
|
nextEl = el.Next()
|
|
p := el.Value
|
|
if p.SendTime.After(now.Add(-maxAge)) {
|
|
break
|
|
}
|
|
delete(h.packetMap, p.PacketNumber)
|
|
h.etcPacketList.Remove(el)
|
|
}
|
|
}
|
|
|
|
func (h *sentPacketHistory) DeclareLost(p *Packet) *Packet {
|
|
el, ok := h.packetMap[p.PacketNumber]
|
|
if !ok {
|
|
return nil
|
|
}
|
|
el.List().Remove(el)
|
|
p.declaredLost = true
|
|
// move it to the correct position in the etc list (based on the packet number)
|
|
for el = h.etcPacketList.Back(); el != nil; el = el.Prev() {
|
|
if el.Value.PacketNumber < p.PacketNumber {
|
|
break
|
|
}
|
|
}
|
|
if el == nil {
|
|
el = h.etcPacketList.PushFront(p)
|
|
} else {
|
|
el = h.etcPacketList.InsertAfter(p, el)
|
|
}
|
|
h.packetMap[p.PacketNumber] = el
|
|
return el.Value
|
|
}
|