mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-05 13:17:36 +03:00
fix connection busy-looping when pacing with a blocked send queue (#4943)
This commit is contained in:
parent
108b6603c8
commit
73108c2a51
2 changed files with 61 additions and 2 deletions
|
@ -637,9 +637,10 @@ runLoop:
|
||||||
}
|
}
|
||||||
|
|
||||||
if s.sendQueue.WouldBlock() {
|
if s.sendQueue.WouldBlock() {
|
||||||
// The send queue is still busy sending out packets.
|
// The send queue is still busy sending out packets. Wait until there's space to enqueue new packets.
|
||||||
// Wait until there's space to enqueue new packets.
|
|
||||||
sendQueueAvailable = s.sendQueue.Available()
|
sendQueueAvailable = s.sendQueue.Available()
|
||||||
|
// Cancel the pacing timer, as we can't send any more packets until the send queue is available again.
|
||||||
|
s.pacingDeadline = time.Time{}
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -652,7 +653,10 @@ runLoop:
|
||||||
break runLoop
|
break runLoop
|
||||||
}
|
}
|
||||||
if s.sendQueue.WouldBlock() {
|
if s.sendQueue.WouldBlock() {
|
||||||
|
// The send queue is still busy sending out packets. Wait until there's space to enqueue new packets.
|
||||||
sendQueueAvailable = s.sendQueue.Available()
|
sendQueueAvailable = s.sendQueue.Available()
|
||||||
|
// Cancel the pacing timer, as we can't send any more packets until the send queue is available again.
|
||||||
|
s.pacingDeadline = time.Time{}
|
||||||
} else {
|
} else {
|
||||||
sendQueueAvailable = nil
|
sendQueueAvailable = nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -1724,6 +1724,61 @@ func TestConnectionPacketPacing(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// When the send queue blocks, we need to reset the pacing timer, otherwise the run loop might busy-loop.
|
||||||
|
// See https://github.com/quic-go/quic-go/pull/4943 for more details.
|
||||||
|
func TestConnectionPacingAndSendQueue(t *testing.T) {
|
||||||
|
mockCtrl := gomock.NewController(t)
|
||||||
|
sph := mockackhandler.NewMockSentPacketHandler(mockCtrl)
|
||||||
|
sender := NewMockSender(mockCtrl)
|
||||||
|
|
||||||
|
tc := newServerTestConnection(t,
|
||||||
|
mockCtrl,
|
||||||
|
nil,
|
||||||
|
false,
|
||||||
|
connectionOptSentPacketHandler(sph),
|
||||||
|
connectionOptSender(sender),
|
||||||
|
connectionOptHandshakeConfirmed(),
|
||||||
|
// set a fixed RTT, so that the idle timeout doesn't interfere with this test
|
||||||
|
connectionOptRTT(10*time.Second),
|
||||||
|
)
|
||||||
|
sender.EXPECT().Run()
|
||||||
|
|
||||||
|
sendQueueAvailable := make(chan struct{})
|
||||||
|
pacingDeadline := time.Now().Add(-time.Millisecond)
|
||||||
|
var counter int
|
||||||
|
// allow exactly one packet to be sent, then become blocked
|
||||||
|
sender.EXPECT().WouldBlock().Return(false)
|
||||||
|
sender.EXPECT().WouldBlock().DoAndReturn(func() bool { counter++; return true }).AnyTimes()
|
||||||
|
sender.EXPECT().Available().Return(sendQueueAvailable).AnyTimes()
|
||||||
|
sph.EXPECT().GetLossDetectionTimeout().Return(time.Now().Add(time.Hour)).AnyTimes()
|
||||||
|
sph.EXPECT().SendMode(gomock.Any()).Return(ackhandler.SendPacingLimited).AnyTimes()
|
||||||
|
sph.EXPECT().TimeUntilSend().Return(pacingDeadline).AnyTimes()
|
||||||
|
sph.EXPECT().ECNMode(gomock.Any()).Return(protocol.ECNNon).AnyTimes()
|
||||||
|
tc.packer.EXPECT().PackAckOnlyPacket(gomock.Any(), gomock.Any(), gomock.Any()).Return(
|
||||||
|
shortHeaderPacket{}, nil, errNothingToPack,
|
||||||
|
)
|
||||||
|
|
||||||
|
errChan := make(chan error, 1)
|
||||||
|
go func() { errChan <- tc.conn.run() }()
|
||||||
|
tc.conn.scheduleSending()
|
||||||
|
|
||||||
|
time.Sleep(scaleDuration(10 * time.Millisecond))
|
||||||
|
|
||||||
|
// test teardown
|
||||||
|
tc.connRunner.EXPECT().Remove(gomock.Any()).AnyTimes()
|
||||||
|
sender.EXPECT().Close()
|
||||||
|
tc.conn.destroy(nil)
|
||||||
|
select {
|
||||||
|
case err := <-errChan:
|
||||||
|
require.NoError(t, err)
|
||||||
|
case <-time.After(time.Second):
|
||||||
|
t.Fatal("timeout")
|
||||||
|
}
|
||||||
|
|
||||||
|
// make sure the run loop didn't do too many iterations
|
||||||
|
require.Less(t, counter, 3)
|
||||||
|
}
|
||||||
|
|
||||||
func TestConnectionIdleTimeout(t *testing.T) {
|
func TestConnectionIdleTimeout(t *testing.T) {
|
||||||
mockCtrl := gomock.NewController(t)
|
mockCtrl := gomock.NewController(t)
|
||||||
sph := mockackhandler.NewMockSentPacketHandler(mockCtrl)
|
sph := mockackhandler.NewMockSentPacketHandler(mockCtrl)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue