bundle FINs into stream frames (sometimes)

fixes #279
This commit is contained in:
Lucas Clemente 2016-08-17 23:11:39 +02:00
parent f14f291032
commit 73c8967302
2 changed files with 50 additions and 8 deletions

View file

@ -1,6 +1,8 @@
package quic package quic
import ( import (
"runtime"
"github.com/lucas-clemente/quic-go/flowcontrol" "github.com/lucas-clemente/quic-go/flowcontrol"
"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"
@ -96,17 +98,24 @@ func (f *streamFramer) maybePopNormalFrames(maxBytes protocol.ByteCount) (res []
} }
data := s.getDataForWriting(maxLen) data := s.getDataForWriting(maxLen)
if data == nil {
if s.shouldSendFin() { // Here, stream.Write() may return in parallel. Afterwards, the user may
frame.FinBit = true // call stream.Close(). We want to pack the FIN into the same frame,
s.sentFin() // so we speculatively allow the other goroutines to run.
res = append(res, frame) // In tests, this increased the percentage of FINs packed into the same
currentLen += frameHeaderBytes + frame.DataLen() // frame from ~20% to ~97%.
frame = &frames.StreamFrame{DataLenPresent: true} runtime.Gosched()
}
shouldSendFin := s.shouldSendFin()
if data == nil && !shouldSendFin {
return true, nil return true, nil
} }
if shouldSendFin {
frame.FinBit = true
s.sentFin()
}
frame.Data = data frame.Data = data
f.flowControlManager.AddBytesSent(s.streamID, protocol.ByteCount(len(data))) f.flowControlManager.AddBytesSent(s.streamID, protocol.ByteCount(len(data)))

View file

@ -2,6 +2,7 @@ package quic
import ( import (
"bytes" "bytes"
"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"
@ -253,6 +254,38 @@ var _ = Describe("Stream Framer", func() {
Expect(fs[0].FinBit).To(BeTrue()) Expect(fs[0].FinBit).To(BeTrue())
Expect(fs[0].Data).To(BeEmpty()) Expect(fs[0].Data).To(BeEmpty())
}) })
It("bundles FINs with data", func() {
// Since this is non-deterministic (see the comment in maybePopNormalFrames),
// we give it a few tries and assert that FINs were packed at least
// some times.
stream1.onData = func() {}
stream1.doneWritingOrErrCond.L = &stream1.mutex
const n = 1000
nFins := 0
for i := 0; i < n; {
go func() {
defer GinkgoRecover()
_, err := stream1.Write([]byte("foobar"))
Expect(err).NotTo(HaveOccurred())
stream1.Close()
}()
time.Sleep(time.Microsecond)
fs := framer.PopStreamFrames(1000)
if len(fs) != 1 {
continue
}
Expect(fs[0].StreamID).To(Equal(stream1.streamID))
if fs[0].FinBit {
nFins++
}
stream1.closed = 0
stream1.finSent = false
stream1.dataForWriting = nil
i++
}
Expect(nFins).To(BeNumerically(">", n/2))
}, 5)
}) })
}) })