don't queue the complete window update, but only the stream ID

This commit is contained in:
Marten Seemann 2017-12-21 17:49:58 +07:00
parent 15ab0ae443
commit d195085f65
13 changed files with 324 additions and 77 deletions

View file

@ -10,31 +10,48 @@ import (
type windowUpdateQueue struct {
mutex sync.Mutex
queue map[protocol.StreamID]protocol.ByteCount
callback func(wire.Frame)
queue map[protocol.StreamID]bool // used as a set
callback func(wire.Frame)
cryptoStream cryptoStreamI
streamGetter streamGetter
}
func newWindowUpdateQueue(cb func(wire.Frame)) *windowUpdateQueue {
func newWindowUpdateQueue(streamGetter streamGetter, cryptoStream cryptoStreamI, cb func(wire.Frame)) *windowUpdateQueue {
return &windowUpdateQueue{
queue: make(map[protocol.StreamID]protocol.ByteCount),
callback: cb,
queue: make(map[protocol.StreamID]bool),
streamGetter: streamGetter,
cryptoStream: cryptoStream,
callback: cb,
}
}
func (q *windowUpdateQueue) Add(stream protocol.StreamID, offset protocol.ByteCount) {
func (q *windowUpdateQueue) Add(id protocol.StreamID) {
q.mutex.Lock()
q.queue[stream] = offset
q.queue[id] = true
q.mutex.Unlock()
}
func (q *windowUpdateQueue) QueueAll() {
q.mutex.Lock()
for stream, offset := range q.queue {
var offset protocol.ByteCount
for id := range q.queue {
if id == q.cryptoStream.StreamID() {
offset = q.cryptoStream.getWindowUpdate()
} else {
str, err := q.streamGetter.GetOrOpenStream(id)
if err != nil || str == nil { // the stream can be nil if it was completed before dequeing the window update
continue
}
offset = str.getWindowUpdate()
}
if offset == 0 { // can happen if we received a final offset, right after queueing the window update
continue
}
q.callback(&wire.MaxStreamDataFrame{
StreamID: stream,
StreamID: id,
ByteOffset: offset,
})
delete(q.queue, stream)
delete(q.queue, id)
}
q.mutex.Unlock()
}