mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
ackhandler: add path probe tracking logic to sent packet history (#4934)
This commit is contained in:
parent
94dc188974
commit
895faf6dc9
3 changed files with 149 additions and 7 deletions
|
@ -22,10 +22,11 @@ type packet struct {
|
||||||
includedInBytesInFlight bool
|
includedInBytesInFlight bool
|
||||||
declaredLost bool
|
declaredLost bool
|
||||||
skippedPacket bool
|
skippedPacket bool
|
||||||
|
isPathProbePacket bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *packet) outstanding() bool {
|
func (p *packet) outstanding() bool {
|
||||||
return !p.declaredLost && !p.skippedPacket && !p.IsPathMTUProbePacket
|
return !p.declaredLost && !p.skippedPacket && !p.IsPathMTUProbePacket && !p.isPathProbePacket
|
||||||
}
|
}
|
||||||
|
|
||||||
var packetPool = sync.Pool{New: func() any { return &packet{} }}
|
var packetPool = sync.Pool{New: func() any { return &packet{} }}
|
||||||
|
|
|
@ -7,7 +7,8 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type sentPacketHistory struct {
|
type sentPacketHistory struct {
|
||||||
packets []*packet
|
packets []*packet
|
||||||
|
pathProbePackets []*packet
|
||||||
|
|
||||||
numOutstanding int
|
numOutstanding int
|
||||||
|
|
||||||
|
@ -32,11 +33,11 @@ func (h *sentPacketHistory) checkSequentialPacketNumberUse(pn protocol.PacketNum
|
||||||
panic("non-sequential packet number use")
|
panic("non-sequential packet number use")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
h.highestPacketNumber = pn
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *sentPacketHistory) SkippedPacket(pn protocol.PacketNumber) {
|
func (h *sentPacketHistory) SkippedPacket(pn protocol.PacketNumber) {
|
||||||
h.checkSequentialPacketNumberUse(pn)
|
h.checkSequentialPacketNumberUse(pn)
|
||||||
h.highestPacketNumber = pn
|
|
||||||
h.packets = append(h.packets, &packet{
|
h.packets = append(h.packets, &packet{
|
||||||
PacketNumber: pn,
|
PacketNumber: pn,
|
||||||
skippedPacket: true,
|
skippedPacket: true,
|
||||||
|
@ -45,7 +46,6 @@ func (h *sentPacketHistory) SkippedPacket(pn protocol.PacketNumber) {
|
||||||
|
|
||||||
func (h *sentPacketHistory) SentNonAckElicitingPacket(pn protocol.PacketNumber) {
|
func (h *sentPacketHistory) SentNonAckElicitingPacket(pn protocol.PacketNumber) {
|
||||||
h.checkSequentialPacketNumberUse(pn)
|
h.checkSequentialPacketNumberUse(pn)
|
||||||
h.highestPacketNumber = pn
|
|
||||||
if len(h.packets) > 0 {
|
if len(h.packets) > 0 {
|
||||||
h.packets = append(h.packets, nil)
|
h.packets = append(h.packets, nil)
|
||||||
}
|
}
|
||||||
|
@ -53,21 +53,37 @@ func (h *sentPacketHistory) SentNonAckElicitingPacket(pn protocol.PacketNumber)
|
||||||
|
|
||||||
func (h *sentPacketHistory) SentAckElicitingPacket(p *packet) {
|
func (h *sentPacketHistory) SentAckElicitingPacket(p *packet) {
|
||||||
h.checkSequentialPacketNumberUse(p.PacketNumber)
|
h.checkSequentialPacketNumberUse(p.PacketNumber)
|
||||||
h.highestPacketNumber = p.PacketNumber
|
|
||||||
h.packets = append(h.packets, p)
|
h.packets = append(h.packets, p)
|
||||||
if p.outstanding() {
|
if p.outstanding() {
|
||||||
h.numOutstanding++
|
h.numOutstanding++
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (h *sentPacketHistory) SentPathProbePacket(p *packet) {
|
||||||
|
h.checkSequentialPacketNumberUse(p.PacketNumber)
|
||||||
|
h.packets = append(h.packets, &packet{
|
||||||
|
PacketNumber: p.PacketNumber,
|
||||||
|
isPathProbePacket: true,
|
||||||
|
})
|
||||||
|
h.pathProbePackets = append(h.pathProbePackets, p)
|
||||||
|
}
|
||||||
|
|
||||||
// Iterate iterates through all packets.
|
// Iterate iterates through all packets.
|
||||||
func (h *sentPacketHistory) Iterate(cb func(*packet) (cont bool)) {
|
func (h *sentPacketHistory) Iterate(cb func(*packet) (cont bool)) {
|
||||||
for _, p := range h.packets {
|
for _, p := range h.packets {
|
||||||
if p == nil {
|
if p == nil {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
cont := cb(p)
|
if cont := cb(p); !cont {
|
||||||
if !cont {
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// IteratePathProbes iterates through all packets.
|
||||||
|
func (h *sentPacketHistory) IteratePathProbes(cb func(*packet) (cont bool)) {
|
||||||
|
for _, p := range h.pathProbePackets {
|
||||||
|
if cont := cb(p); !cont {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -86,6 +102,14 @@ func (h *sentPacketHistory) FirstOutstanding() *packet {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FirstOutstandingPathProbe returns the first outstanding path probe packet
|
||||||
|
func (h *sentPacketHistory) FirstOutstandingPathProbe() *packet {
|
||||||
|
if len(h.pathProbePackets) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return h.pathProbePackets[0]
|
||||||
|
}
|
||||||
|
|
||||||
func (h *sentPacketHistory) Len() int {
|
func (h *sentPacketHistory) Len() int {
|
||||||
return len(h.packets)
|
return len(h.packets)
|
||||||
}
|
}
|
||||||
|
@ -121,6 +145,27 @@ func (h *sentPacketHistory) Remove(pn protocol.PacketNumber) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// RemovePathProbe removes a path probe packet.
|
||||||
|
// It scales O(N), but that's ok, since we don't expect to send many path probe packets.
|
||||||
|
// It is not valid to call this function in IteratePathProbes.
|
||||||
|
func (h *sentPacketHistory) RemovePathProbe(pn protocol.PacketNumber) *packet {
|
||||||
|
var packetToDelete *packet
|
||||||
|
idx := -1
|
||||||
|
for i, p := range h.pathProbePackets {
|
||||||
|
if p.PacketNumber == pn {
|
||||||
|
packetToDelete = p
|
||||||
|
idx = i
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if idx != -1 {
|
||||||
|
// don't use slices.Delete, because it zeros the deleted element
|
||||||
|
copy(h.pathProbePackets[idx:], h.pathProbePackets[idx+1:])
|
||||||
|
h.pathProbePackets = h.pathProbePackets[:len(h.pathProbePackets)-1]
|
||||||
|
}
|
||||||
|
return packetToDelete
|
||||||
|
}
|
||||||
|
|
||||||
// getIndex gets the index of packet p in the packets slice.
|
// getIndex gets the index of packet p in the packets slice.
|
||||||
func (h *sentPacketHistory) getIndex(p protocol.PacketNumber) (int, bool) {
|
func (h *sentPacketHistory) getIndex(p protocol.PacketNumber) (int, bool) {
|
||||||
if len(h.packets) == 0 {
|
if len(h.packets) == 0 {
|
||||||
|
@ -141,6 +186,10 @@ func (h *sentPacketHistory) HasOutstandingPackets() bool {
|
||||||
return h.numOutstanding > 0
|
return h.numOutstanding > 0
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (h *sentPacketHistory) HasOutstandingPathProbes() bool {
|
||||||
|
return len(h.pathProbePackets) > 0
|
||||||
|
}
|
||||||
|
|
||||||
// delete all nil entries at the beginning of the packets slice
|
// delete all nil entries at the beginning of the packets slice
|
||||||
func (h *sentPacketHistory) cleanupStart() {
|
func (h *sentPacketHistory) cleanupStart() {
|
||||||
for i, p := range h.packets {
|
for i, p := range h.packets {
|
||||||
|
|
|
@ -204,3 +204,95 @@ func TestSentPacketHistoryDeleteWhileIterating(t *testing.T) {
|
||||||
require.Equal(t, []protocol.PacketNumber{1, 3, 5}, hist.getPacketNumbers())
|
require.Equal(t, []protocol.PacketNumber{1, 3, 5}, hist.getPacketNumbers())
|
||||||
require.Equal(t, []protocol.PacketNumber{2}, hist.getSkippedPacketNumbers())
|
require.Equal(t, []protocol.PacketNumber{2}, hist.getSkippedPacketNumbers())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestSentPacketHistoryPathProbes(t *testing.T) {
|
||||||
|
hist := newSentPacketHistory(true)
|
||||||
|
hist.SentAckElicitingPacket(&packet{PacketNumber: 0})
|
||||||
|
hist.SentAckElicitingPacket(&packet{PacketNumber: 1})
|
||||||
|
hist.SentPathProbePacket(&packet{PacketNumber: 2})
|
||||||
|
hist.SentAckElicitingPacket(&packet{PacketNumber: 3})
|
||||||
|
hist.SentAckElicitingPacket(&packet{PacketNumber: 4})
|
||||||
|
hist.SentPathProbePacket(&packet{PacketNumber: 5})
|
||||||
|
|
||||||
|
getPacketsInHistory := func(t *testing.T) []protocol.PacketNumber {
|
||||||
|
t.Helper()
|
||||||
|
var pns []protocol.PacketNumber
|
||||||
|
hist.Iterate(func(p *packet) bool {
|
||||||
|
pns = append(pns, p.PacketNumber)
|
||||||
|
switch p.PacketNumber {
|
||||||
|
case 2, 5:
|
||||||
|
require.True(t, p.isPathProbePacket)
|
||||||
|
default:
|
||||||
|
require.False(t, p.isPathProbePacket)
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
return pns
|
||||||
|
}
|
||||||
|
|
||||||
|
getPacketsInPathProbeHistory := func(t *testing.T) []protocol.PacketNumber {
|
||||||
|
t.Helper()
|
||||||
|
var pns []protocol.PacketNumber
|
||||||
|
hist.IteratePathProbes(func(p *packet) bool {
|
||||||
|
pns = append(pns, p.PacketNumber)
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
return pns
|
||||||
|
}
|
||||||
|
|
||||||
|
require.Equal(t, []protocol.PacketNumber{0, 1, 2, 3, 4, 5}, getPacketsInHistory(t))
|
||||||
|
require.Equal(t, []protocol.PacketNumber{2, 5}, getPacketsInPathProbeHistory(t))
|
||||||
|
|
||||||
|
// Removing packets from the regular packet history might happen before the path probe
|
||||||
|
// is declared lost, as the original path might have a smaller RTT than the path timeout.
|
||||||
|
// Therefore, the path probe packet is not removed from the path probe history.
|
||||||
|
require.NoError(t, hist.Remove(0))
|
||||||
|
require.NoError(t, hist.Remove(1))
|
||||||
|
require.NoError(t, hist.Remove(2))
|
||||||
|
require.NoError(t, hist.Remove(3))
|
||||||
|
require.Equal(t, []protocol.PacketNumber{4, 5}, getPacketsInHistory(t))
|
||||||
|
require.Equal(t, []protocol.PacketNumber{2, 5}, getPacketsInPathProbeHistory(t))
|
||||||
|
require.True(t, hist.HasOutstandingPackets())
|
||||||
|
require.True(t, hist.HasOutstandingPathProbes())
|
||||||
|
firstOutstanding := hist.FirstOutstanding()
|
||||||
|
require.NotNil(t, firstOutstanding)
|
||||||
|
require.Equal(t, protocol.PacketNumber(4), firstOutstanding.PacketNumber)
|
||||||
|
firstOutStandingPathProbe := hist.FirstOutstandingPathProbe()
|
||||||
|
require.NotNil(t, firstOutStandingPathProbe)
|
||||||
|
require.Equal(t, protocol.PacketNumber(2), firstOutStandingPathProbe.PacketNumber)
|
||||||
|
|
||||||
|
hist.RemovePathProbe(2)
|
||||||
|
require.Equal(t, []protocol.PacketNumber{4, 5}, getPacketsInHistory(t))
|
||||||
|
require.Equal(t, []protocol.PacketNumber{5}, getPacketsInPathProbeHistory(t))
|
||||||
|
require.True(t, hist.HasOutstandingPathProbes())
|
||||||
|
firstOutStandingPathProbe = hist.FirstOutstandingPathProbe()
|
||||||
|
require.NotNil(t, firstOutStandingPathProbe)
|
||||||
|
require.Equal(t, protocol.PacketNumber(5), firstOutStandingPathProbe.PacketNumber)
|
||||||
|
|
||||||
|
hist.RemovePathProbe(5)
|
||||||
|
require.Equal(t, []protocol.PacketNumber{4, 5}, getPacketsInHistory(t))
|
||||||
|
require.Empty(t, getPacketsInPathProbeHistory(t))
|
||||||
|
require.True(t, hist.HasOutstandingPackets())
|
||||||
|
require.False(t, hist.HasOutstandingPathProbes())
|
||||||
|
require.Nil(t, hist.FirstOutstandingPathProbe())
|
||||||
|
|
||||||
|
require.NoError(t, hist.Remove(4))
|
||||||
|
require.NoError(t, hist.Remove(5))
|
||||||
|
require.Empty(t, getPacketsInHistory(t))
|
||||||
|
require.False(t, hist.HasOutstandingPackets())
|
||||||
|
require.Nil(t, hist.FirstOutstanding())
|
||||||
|
|
||||||
|
// path probe packets are considered outstanding
|
||||||
|
hist.SentPathProbePacket(&packet{PacketNumber: 6})
|
||||||
|
require.False(t, hist.HasOutstandingPackets())
|
||||||
|
require.True(t, hist.HasOutstandingPathProbes())
|
||||||
|
firstOutStandingPathProbe = hist.FirstOutstandingPathProbe()
|
||||||
|
require.NotNil(t, firstOutStandingPathProbe)
|
||||||
|
require.Equal(t, protocol.PacketNumber(6), firstOutStandingPathProbe.PacketNumber)
|
||||||
|
|
||||||
|
hist.RemovePathProbe(6)
|
||||||
|
require.False(t, hist.HasOutstandingPackets())
|
||||||
|
require.Nil(t, hist.FirstOutstanding())
|
||||||
|
require.False(t, hist.HasOutstandingPathProbes())
|
||||||
|
require.Nil(t, hist.FirstOutstandingPathProbe())
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue