mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
parent
f14f291032
commit
73c8967302
2 changed files with 50 additions and 8 deletions
|
@ -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)))
|
||||||
|
|
||||||
|
|
|
@ -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)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue