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 }