mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-03 20:27:35 +03:00
180 lines
4.2 KiB
Go
180 lines
4.2 KiB
Go
package quic
|
|
|
|
import (
|
|
"errors"
|
|
|
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
|
"github.com/lucas-clemente/quic-go/internal/utils"
|
|
)
|
|
|
|
type frameSorterEntry struct {
|
|
Data []byte
|
|
DoneCb func()
|
|
}
|
|
|
|
type frameSorter struct {
|
|
queue map[protocol.ByteCount]frameSorterEntry
|
|
readPos protocol.ByteCount
|
|
gaps *utils.ByteIntervalList
|
|
}
|
|
|
|
var errDuplicateStreamData = errors.New("duplicate stream data")
|
|
|
|
func newFrameSorter() *frameSorter {
|
|
s := frameSorter{
|
|
gaps: utils.NewByteIntervalList(),
|
|
queue: make(map[protocol.ByteCount]frameSorterEntry),
|
|
}
|
|
s.gaps.PushFront(utils.ByteInterval{Start: 0, End: protocol.MaxByteCount})
|
|
return &s
|
|
}
|
|
|
|
func (s *frameSorter) Push(data []byte, offset protocol.ByteCount, doneCb func()) error {
|
|
err := s.push(data, offset, doneCb)
|
|
if err == errDuplicateStreamData {
|
|
if doneCb != nil {
|
|
doneCb()
|
|
}
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (s *frameSorter) push(data []byte, offset protocol.ByteCount, doneCb func()) error {
|
|
if len(data) == 0 {
|
|
return errDuplicateStreamData
|
|
}
|
|
|
|
if oldEntry, ok := s.queue[offset]; ok {
|
|
if len(data) <= len(oldEntry.Data) {
|
|
return errDuplicateStreamData
|
|
}
|
|
// The data we currently have is shorter than the new data.
|
|
// Replace it.
|
|
if oldEntry.DoneCb != nil {
|
|
oldEntry.DoneCb()
|
|
}
|
|
s.queue[offset] = frameSorterEntry{Data: data, DoneCb: doneCb}
|
|
}
|
|
|
|
start := offset
|
|
end := offset + protocol.ByteCount(len(data))
|
|
|
|
// skip all gaps that are before this stream frame
|
|
var gap *utils.ByteIntervalElement
|
|
for gap = s.gaps.Front(); gap != nil; gap = gap.Next() {
|
|
// the frame is a duplicate. Ignore it
|
|
if end <= gap.Value.Start {
|
|
return errDuplicateStreamData
|
|
}
|
|
if end > gap.Value.Start && start <= gap.Value.End {
|
|
break
|
|
}
|
|
}
|
|
|
|
if gap == nil {
|
|
return errors.New("StreamFrameSorter BUG: no gap found")
|
|
}
|
|
|
|
var wasCut bool
|
|
if start < gap.Value.Start {
|
|
add := gap.Value.Start - start
|
|
offset += add
|
|
start += add
|
|
data = data[add:]
|
|
wasCut = true
|
|
}
|
|
|
|
// find the highest gaps whose Start lies before the end of the frame
|
|
endGap := gap
|
|
for end >= endGap.Value.End {
|
|
nextEndGap := endGap.Next()
|
|
if nextEndGap == nil {
|
|
return errors.New("StreamFrameSorter BUG: no end gap found")
|
|
}
|
|
if endGap != gap {
|
|
s.gaps.Remove(endGap)
|
|
}
|
|
if end < nextEndGap.Value.Start {
|
|
break
|
|
}
|
|
// delete queued frames completely covered by the current frame
|
|
end := endGap.Value.End
|
|
if end != offset {
|
|
if cb := s.queue[end].DoneCb; cb != nil {
|
|
cb()
|
|
}
|
|
delete(s.queue, end)
|
|
}
|
|
endGap = nextEndGap
|
|
}
|
|
|
|
if end > endGap.Value.End {
|
|
cutLen := end - endGap.Value.End
|
|
len := protocol.ByteCount(len(data)) - cutLen
|
|
end -= cutLen
|
|
data = data[:len]
|
|
wasCut = true
|
|
}
|
|
|
|
if start == gap.Value.Start {
|
|
if end >= gap.Value.End {
|
|
// the frame completely fills this gap
|
|
// delete the gap
|
|
s.gaps.Remove(gap)
|
|
}
|
|
if end < endGap.Value.End {
|
|
// the frame covers the beginning of the gap
|
|
// adjust the Start value to shrink the gap
|
|
endGap.Value.Start = end
|
|
}
|
|
} else if end == endGap.Value.End {
|
|
// the frame covers the end of the gap
|
|
// adjust the End value to shrink the gap
|
|
gap.Value.End = start
|
|
} else {
|
|
if gap == endGap {
|
|
// the frame lies within the current gap, splitting it into two
|
|
// insert a new gap and adjust the current one
|
|
intv := utils.ByteInterval{Start: end, End: gap.Value.End}
|
|
s.gaps.InsertAfter(intv, gap)
|
|
gap.Value.End = start
|
|
} else {
|
|
gap.Value.End = start
|
|
endGap.Value.Start = end
|
|
}
|
|
}
|
|
|
|
if s.gaps.Len() > protocol.MaxStreamFrameSorterGaps {
|
|
return errors.New("too many gaps in received data")
|
|
}
|
|
|
|
if wasCut && len(data) < protocol.MinStreamFrameBufferSize {
|
|
newData := make([]byte, len(data))
|
|
copy(newData, data)
|
|
data = newData
|
|
if doneCb != nil {
|
|
doneCb()
|
|
doneCb = nil
|
|
}
|
|
}
|
|
|
|
s.queue[offset] = frameSorterEntry{Data: data, DoneCb: doneCb}
|
|
return nil
|
|
}
|
|
|
|
func (s *frameSorter) Pop() (protocol.ByteCount, []byte, func()) {
|
|
entry, ok := s.queue[s.readPos]
|
|
if !ok {
|
|
return s.readPos, nil, nil
|
|
}
|
|
delete(s.queue, s.readPos)
|
|
offset := s.readPos
|
|
s.readPos += protocol.ByteCount(len(entry.Data))
|
|
return offset, entry.Data, entry.DoneCb
|
|
}
|
|
|
|
// HasMoreData says if there is any more data queued at *any* offset.
|
|
func (s *frameSorter) HasMoreData() bool {
|
|
return len(s.queue) > 0
|
|
}
|