forked from quic-go/quic-go
save timestamp when a packet arrives in ReceivedPacketHandler
work towards #10, #81
This commit is contained in:
@@ -2,6 +2,7 @@ package ackhandler
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/frames"
|
"github.com/lucas-clemente/quic-go/frames"
|
||||||
"github.com/lucas-clemente/quic-go/protocol"
|
"github.com/lucas-clemente/quic-go/protocol"
|
||||||
@@ -10,18 +11,23 @@ import (
|
|||||||
// ErrDuplicatePacket occurres when a duplicate packet is received
|
// ErrDuplicatePacket occurres when a duplicate packet is received
|
||||||
var ErrDuplicatePacket = errors.New("ReceivedPacketHandler: Duplicate Packet")
|
var ErrDuplicatePacket = errors.New("ReceivedPacketHandler: Duplicate Packet")
|
||||||
|
|
||||||
|
type packetHistoryEntry struct {
|
||||||
|
EntropyBit bool
|
||||||
|
TimeReceived time.Time
|
||||||
|
}
|
||||||
|
|
||||||
type receivedPacketHandler struct {
|
type receivedPacketHandler struct {
|
||||||
highestInOrderObserved protocol.PacketNumber
|
highestInOrderObserved protocol.PacketNumber
|
||||||
highestInOrderObservedEntropy EntropyAccumulator
|
highestInOrderObservedEntropy EntropyAccumulator
|
||||||
largestObserved protocol.PacketNumber
|
largestObserved protocol.PacketNumber
|
||||||
packetHistory map[protocol.PacketNumber]bool // the bool is the EntropyBit of the packet
|
packetHistory map[protocol.PacketNumber]packetHistoryEntry
|
||||||
stateChanged bool // has an ACK for this state already been sent? Will be set to false every time a new packet arrives, and to false every time an ACK is sent
|
stateChanged bool // has an ACK for this state already been sent? Will be set to false every time a new packet arrives, and to false every time an ACK is sent
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewReceivedPacketHandler creates a new receivedPacketHandler
|
// NewReceivedPacketHandler creates a new receivedPacketHandler
|
||||||
func NewReceivedPacketHandler() ReceivedPacketHandler {
|
func NewReceivedPacketHandler() ReceivedPacketHandler {
|
||||||
return &receivedPacketHandler{
|
return &receivedPacketHandler{
|
||||||
packetHistory: make(map[protocol.PacketNumber]bool),
|
packetHistory: make(map[protocol.PacketNumber]packetHistoryEntry),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -29,7 +35,8 @@ func (h *receivedPacketHandler) ReceivedPacket(packetNumber protocol.PacketNumbe
|
|||||||
if packetNumber == 0 {
|
if packetNumber == 0 {
|
||||||
return errors.New("Invalid packet number")
|
return errors.New("Invalid packet number")
|
||||||
}
|
}
|
||||||
if packetNumber <= h.highestInOrderObserved || h.packetHistory[packetNumber] {
|
_, ok := h.packetHistory[packetNumber]
|
||||||
|
if packetNumber <= h.highestInOrderObserved || ok {
|
||||||
return ErrDuplicatePacket
|
return ErrDuplicatePacket
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -44,7 +51,10 @@ func (h *receivedPacketHandler) ReceivedPacket(packetNumber protocol.PacketNumbe
|
|||||||
h.highestInOrderObservedEntropy.Add(packetNumber, entropyBit)
|
h.highestInOrderObservedEntropy.Add(packetNumber, entropyBit)
|
||||||
}
|
}
|
||||||
|
|
||||||
h.packetHistory[packetNumber] = entropyBit
|
h.packetHistory[packetNumber] = packetHistoryEntry{
|
||||||
|
EntropyBit: entropyBit,
|
||||||
|
TimeReceived: time.Now(),
|
||||||
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -68,7 +78,7 @@ func (h *receivedPacketHandler) getNackRanges() ([]frames.NackRange, EntropyAccu
|
|||||||
inRange := false
|
inRange := false
|
||||||
entropy := h.highestInOrderObservedEntropy
|
entropy := h.highestInOrderObservedEntropy
|
||||||
for i := h.largestObserved; i > h.highestInOrderObserved; i-- {
|
for i := h.largestObserved; i > h.highestInOrderObserved; i-- {
|
||||||
entropyBit, ok := h.packetHistory[i]
|
p, ok := h.packetHistory[i]
|
||||||
if !ok {
|
if !ok {
|
||||||
if !inRange {
|
if !inRange {
|
||||||
r := frames.NackRange{
|
r := frames.NackRange{
|
||||||
@@ -82,7 +92,7 @@ func (h *receivedPacketHandler) getNackRanges() ([]frames.NackRange, EntropyAccu
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
inRange = false
|
inRange = false
|
||||||
entropy.Add(i, entropyBit)
|
entropy.Add(i, p.EntropyBit)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return ranges, entropy
|
return ranges, entropy
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
package ackhandler
|
package ackhandler
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/lucas-clemente/quic-go/frames"
|
"github.com/lucas-clemente/quic-go/frames"
|
||||||
"github.com/lucas-clemente/quic-go/protocol"
|
"github.com/lucas-clemente/quic-go/protocol"
|
||||||
|
|
||||||
@@ -19,7 +21,7 @@ var _ = Describe("receivedPacketHandler", func() {
|
|||||||
expectedEntropy = EntropyAccumulator(0)
|
expectedEntropy = EntropyAccumulator(0)
|
||||||
})
|
})
|
||||||
|
|
||||||
Context("accepting and rejecting packets", func() {
|
Context("accepting packets", func() {
|
||||||
It("handles a packet that arrives late", func() {
|
It("handles a packet that arrives late", func() {
|
||||||
err := handler.ReceivedPacket(protocol.PacketNumber(1), false)
|
err := handler.ReceivedPacket(protocol.PacketNumber(1), false)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
@@ -27,8 +29,9 @@ var _ = Describe("receivedPacketHandler", func() {
|
|||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
err = handler.ReceivedPacket(protocol.PacketNumber(2), false)
|
err = handler.ReceivedPacket(protocol.PacketNumber(2), false)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
nackRanges, _ := handler.getNackRanges()
|
Expect(handler.packetHistory).To(HaveKey(protocol.PacketNumber(1)))
|
||||||
Expect(len(nackRanges)).To(Equal(0))
|
Expect(handler.packetHistory).To(HaveKey(protocol.PacketNumber(2)))
|
||||||
|
Expect(handler.packetHistory).To(HaveKey(protocol.PacketNumber(3)))
|
||||||
})
|
})
|
||||||
|
|
||||||
It("rejects a duplicate package with PacketNumber equal to LargestObserved", func() {
|
It("rejects a duplicate package with PacketNumber equal to LargestObserved", func() {
|
||||||
@@ -50,6 +53,13 @@ var _ = Describe("receivedPacketHandler", func() {
|
|||||||
Expect(err).To(HaveOccurred())
|
Expect(err).To(HaveOccurred())
|
||||||
Expect(err).To(Equal(ErrDuplicatePacket))
|
Expect(err).To(Equal(ErrDuplicatePacket))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("saves the time when each packet arrived", func() {
|
||||||
|
err := handler.ReceivedPacket(protocol.PacketNumber(3), false)
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
Expect(handler.packetHistory).To(HaveKey(protocol.PacketNumber(3)))
|
||||||
|
Expect(handler.packetHistory[3].TimeReceived).To(BeTemporally("~", time.Now(), 10*time.Millisecond))
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
Context("Entropy calculation", func() {
|
Context("Entropy calculation", func() {
|
||||||
|
|||||||
Reference in New Issue
Block a user