mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 04:37:36 +03:00
pack packets into large buffers when GSO is available
This commit is contained in:
parent
628ba87727
commit
5b5ffa942b
16 changed files with 339 additions and 186 deletions
|
@ -3,6 +3,8 @@ package quic
|
|||
import (
|
||||
"errors"
|
||||
|
||||
"github.com/quic-go/quic-go/internal/protocol"
|
||||
|
||||
"github.com/golang/mock/gomock"
|
||||
. "github.com/onsi/ginkgo/v2"
|
||||
. "github.com/onsi/gomega"
|
||||
|
@ -26,10 +28,10 @@ var _ = Describe("Send Queue", func() {
|
|||
|
||||
It("sends a packet", func() {
|
||||
p := getPacket([]byte("foobar"))
|
||||
q.Send(p)
|
||||
q.Send(p, 10) // make sure the packet size is passed through to the conn
|
||||
|
||||
written := make(chan struct{})
|
||||
c.EXPECT().Write([]byte("foobar")).Do(func([]byte) { close(written) })
|
||||
c.EXPECT().Write([]byte("foobar"), protocol.ByteCount(10)).Do(func([]byte, protocol.ByteCount) { close(written) })
|
||||
done := make(chan struct{})
|
||||
go func() {
|
||||
defer GinkgoRecover()
|
||||
|
@ -45,19 +47,19 @@ var _ = Describe("Send Queue", func() {
|
|||
It("panics when Send() is called although there's no space in the queue", func() {
|
||||
for i := 0; i < sendQueueCapacity; i++ {
|
||||
Expect(q.WouldBlock()).To(BeFalse())
|
||||
q.Send(getPacket([]byte("foobar")))
|
||||
q.Send(getPacket([]byte("foobar")), 6)
|
||||
}
|
||||
Expect(q.WouldBlock()).To(BeTrue())
|
||||
Expect(func() { q.Send(getPacket([]byte("raboof"))) }).To(Panic())
|
||||
Expect(func() { q.Send(getPacket([]byte("raboof")), 6) }).To(Panic())
|
||||
})
|
||||
|
||||
It("signals when sending is possible again", func() {
|
||||
Expect(q.WouldBlock()).To(BeFalse())
|
||||
q.Send(getPacket([]byte("foobar1")))
|
||||
q.Send(getPacket([]byte("foobar1")), 6)
|
||||
Consistently(q.Available()).ShouldNot(Receive())
|
||||
|
||||
// now start sending out packets. This should free up queue space.
|
||||
c.EXPECT().Write(gomock.Any()).MinTimes(1).MaxTimes(2)
|
||||
c.EXPECT().Write(gomock.Any(), gomock.Any()).MinTimes(1).MaxTimes(2)
|
||||
done := make(chan struct{})
|
||||
go func() {
|
||||
defer GinkgoRecover()
|
||||
|
@ -67,7 +69,7 @@ var _ = Describe("Send Queue", func() {
|
|||
|
||||
Eventually(q.Available()).Should(Receive())
|
||||
Expect(q.WouldBlock()).To(BeFalse())
|
||||
Expect(func() { q.Send(getPacket([]byte("foobar2"))) }).ToNot(Panic())
|
||||
Expect(func() { q.Send(getPacket([]byte("foobar2")), 7) }).ToNot(Panic())
|
||||
|
||||
q.Close()
|
||||
Eventually(done).Should(BeClosed())
|
||||
|
@ -77,7 +79,7 @@ var _ = Describe("Send Queue", func() {
|
|||
write := make(chan struct{}, 1)
|
||||
written := make(chan struct{}, 100)
|
||||
// now start sending out packets. This should free up queue space.
|
||||
c.EXPECT().Write(gomock.Any()).DoAndReturn(func(b []byte) error {
|
||||
c.EXPECT().Write(gomock.Any(), gomock.Any()).DoAndReturn(func([]byte, protocol.ByteCount) error {
|
||||
written <- struct{}{}
|
||||
<-write
|
||||
return nil
|
||||
|
@ -92,19 +94,19 @@ var _ = Describe("Send Queue", func() {
|
|||
close(done)
|
||||
}()
|
||||
|
||||
q.Send(getPacket([]byte("foobar")))
|
||||
q.Send(getPacket([]byte("foobar")), 6)
|
||||
<-written
|
||||
|
||||
// now fill up the send queue
|
||||
for i := 0; i < sendQueueCapacity; i++ {
|
||||
Expect(q.WouldBlock()).To(BeFalse())
|
||||
q.Send(getPacket([]byte("foobar")))
|
||||
q.Send(getPacket([]byte("foobar")), 6)
|
||||
}
|
||||
// One more packet is queued when it's picked up by Run and written to the connection.
|
||||
// In this test, it's blocked on write channel in the mocked Write call.
|
||||
<-written
|
||||
Eventually(q.WouldBlock()).Should(BeFalse())
|
||||
q.Send(getPacket([]byte("foobar")))
|
||||
q.Send(getPacket([]byte("foobar")), 6)
|
||||
|
||||
Expect(q.WouldBlock()).To(BeTrue())
|
||||
Consistently(q.Available()).ShouldNot(Receive())
|
||||
|
@ -130,15 +132,15 @@ var _ = Describe("Send Queue", func() {
|
|||
|
||||
// the run loop exits if there is a write error
|
||||
testErr := errors.New("test error")
|
||||
c.EXPECT().Write(gomock.Any()).Return(testErr)
|
||||
q.Send(getPacket([]byte("foobar")))
|
||||
c.EXPECT().Write(gomock.Any(), gomock.Any()).Return(testErr)
|
||||
q.Send(getPacket([]byte("foobar")), 6)
|
||||
Eventually(done).Should(BeClosed())
|
||||
|
||||
sent := make(chan struct{})
|
||||
go func() {
|
||||
defer GinkgoRecover()
|
||||
q.Send(getPacket([]byte("raboof")))
|
||||
q.Send(getPacket([]byte("quux")))
|
||||
q.Send(getPacket([]byte("raboof")), 6)
|
||||
q.Send(getPacket([]byte("quux")), 4)
|
||||
close(sent)
|
||||
}()
|
||||
|
||||
|
@ -147,7 +149,7 @@ var _ = Describe("Send Queue", func() {
|
|||
|
||||
It("blocks Close() until the packet has been sent out", func() {
|
||||
written := make(chan []byte)
|
||||
c.EXPECT().Write(gomock.Any()).Do(func(p []byte) { written <- p })
|
||||
c.EXPECT().Write(gomock.Any(), gomock.Any()).Do(func(p []byte, _ protocol.ByteCount) { written <- p })
|
||||
done := make(chan struct{})
|
||||
go func() {
|
||||
defer GinkgoRecover()
|
||||
|
@ -155,7 +157,7 @@ var _ = Describe("Send Queue", func() {
|
|||
close(done)
|
||||
}()
|
||||
|
||||
q.Send(getPacket([]byte("foobar")))
|
||||
q.Send(getPacket([]byte("foobar")), 6)
|
||||
|
||||
closed := make(chan struct{})
|
||||
go func() {
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue