mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 12:47:36 +03:00
don't enqueue stream for sending on reordered MAX_STREAM_DATA frames (#4269)
This commit is contained in:
parent
07ec3245bd
commit
198de32ef6
7 changed files with 47 additions and 22 deletions
|
@ -48,10 +48,12 @@ func (c *baseFlowController) AddBytesSent(n protocol.ByteCount) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateSendWindow is called after receiving a MAX_{STREAM_}DATA frame.
|
// UpdateSendWindow is called after receiving a MAX_{STREAM_}DATA frame.
|
||||||
func (c *baseFlowController) UpdateSendWindow(offset protocol.ByteCount) {
|
func (c *baseFlowController) UpdateSendWindow(offset protocol.ByteCount) (updated bool) {
|
||||||
if offset > c.sendWindow {
|
if offset > c.sendWindow {
|
||||||
c.sendWindow = offset
|
c.sendWindow = offset
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *baseFlowController) sendWindowSize() protocol.ByteCount {
|
func (c *baseFlowController) sendWindowSize() protocol.ByteCount {
|
||||||
|
|
|
@ -59,9 +59,9 @@ var _ = Describe("Base Flow controller", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("does not decrease the flow control window", func() {
|
It("does not decrease the flow control window", func() {
|
||||||
controller.UpdateSendWindow(20)
|
Expect(controller.UpdateSendWindow(20)).To(BeTrue())
|
||||||
Expect(controller.sendWindowSize()).To(Equal(protocol.ByteCount(20)))
|
Expect(controller.sendWindowSize()).To(Equal(protocol.ByteCount(20)))
|
||||||
controller.UpdateSendWindow(10)
|
Expect(controller.UpdateSendWindow(10)).To(BeFalse())
|
||||||
Expect(controller.sendWindowSize()).To(Equal(protocol.ByteCount(20)))
|
Expect(controller.sendWindowSize()).To(Equal(protocol.ByteCount(20)))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@ import "github.com/quic-go/quic-go/internal/protocol"
|
||||||
type flowController interface {
|
type flowController interface {
|
||||||
// for sending
|
// for sending
|
||||||
SendWindowSize() protocol.ByteCount
|
SendWindowSize() protocol.ByteCount
|
||||||
UpdateSendWindow(protocol.ByteCount)
|
UpdateSendWindow(protocol.ByteCount) (updated bool)
|
||||||
AddBytesSent(protocol.ByteCount)
|
AddBytesSent(protocol.ByteCount)
|
||||||
// for receiving
|
// for receiving
|
||||||
AddBytesRead(protocol.ByteCount)
|
AddBytesRead(protocol.ByteCount)
|
||||||
|
@ -16,12 +16,11 @@ type flowController interface {
|
||||||
// A StreamFlowController is a flow controller for a QUIC stream.
|
// A StreamFlowController is a flow controller for a QUIC stream.
|
||||||
type StreamFlowController interface {
|
type StreamFlowController interface {
|
||||||
flowController
|
flowController
|
||||||
// for receiving
|
// UpdateHighestReceived is called when a new highest offset is received
|
||||||
// UpdateHighestReceived should be called when a new highest offset is received
|
|
||||||
// final has to be to true if this is the final offset of the stream,
|
// final has to be to true if this is the final offset of the stream,
|
||||||
// as contained in a STREAM frame with FIN bit, and the RESET_STREAM frame
|
// as contained in a STREAM frame with FIN bit, and the RESET_STREAM frame
|
||||||
UpdateHighestReceived(offset protocol.ByteCount, final bool) error
|
UpdateHighestReceived(offset protocol.ByteCount, final bool) error
|
||||||
// Abandon should be called when reading from the stream is aborted early,
|
// Abandon is called when reading from the stream is aborted early,
|
||||||
// and there won't be any further calls to AddBytesRead.
|
// and there won't be any further calls to AddBytesRead.
|
||||||
Abandon()
|
Abandon()
|
||||||
}
|
}
|
||||||
|
|
|
@ -264,9 +264,11 @@ func (c *ConnectionFlowControllerSendWindowSizeCall) DoAndReturn(f func() protoc
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateSendWindow mocks base method.
|
// UpdateSendWindow mocks base method.
|
||||||
func (m *MockConnectionFlowController) UpdateSendWindow(arg0 protocol.ByteCount) {
|
func (m *MockConnectionFlowController) UpdateSendWindow(arg0 protocol.ByteCount) bool {
|
||||||
m.ctrl.T.Helper()
|
m.ctrl.T.Helper()
|
||||||
m.ctrl.Call(m, "UpdateSendWindow", arg0)
|
ret := m.ctrl.Call(m, "UpdateSendWindow", arg0)
|
||||||
|
ret0, _ := ret[0].(bool)
|
||||||
|
return ret0
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateSendWindow indicates an expected call of UpdateSendWindow.
|
// UpdateSendWindow indicates an expected call of UpdateSendWindow.
|
||||||
|
@ -282,19 +284,19 @@ type ConnectionFlowControllerUpdateSendWindowCall struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return rewrite *gomock.Call.Return
|
// Return rewrite *gomock.Call.Return
|
||||||
func (c *ConnectionFlowControllerUpdateSendWindowCall) Return() *ConnectionFlowControllerUpdateSendWindowCall {
|
func (c *ConnectionFlowControllerUpdateSendWindowCall) Return(arg0 bool) *ConnectionFlowControllerUpdateSendWindowCall {
|
||||||
c.Call = c.Call.Return()
|
c.Call = c.Call.Return(arg0)
|
||||||
return c
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
// Do rewrite *gomock.Call.Do
|
// Do rewrite *gomock.Call.Do
|
||||||
func (c *ConnectionFlowControllerUpdateSendWindowCall) Do(f func(protocol.ByteCount)) *ConnectionFlowControllerUpdateSendWindowCall {
|
func (c *ConnectionFlowControllerUpdateSendWindowCall) Do(f func(protocol.ByteCount) bool) *ConnectionFlowControllerUpdateSendWindowCall {
|
||||||
c.Call = c.Call.Do(f)
|
c.Call = c.Call.Do(f)
|
||||||
return c
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
// DoAndReturn rewrite *gomock.Call.DoAndReturn
|
// DoAndReturn rewrite *gomock.Call.DoAndReturn
|
||||||
func (c *ConnectionFlowControllerUpdateSendWindowCall) DoAndReturn(f func(protocol.ByteCount)) *ConnectionFlowControllerUpdateSendWindowCall {
|
func (c *ConnectionFlowControllerUpdateSendWindowCall) DoAndReturn(f func(protocol.ByteCount) bool) *ConnectionFlowControllerUpdateSendWindowCall {
|
||||||
c.Call = c.Call.DoAndReturn(f)
|
c.Call = c.Call.DoAndReturn(f)
|
||||||
return c
|
return c
|
||||||
}
|
}
|
||||||
|
|
|
@ -300,9 +300,11 @@ func (c *StreamFlowControllerUpdateHighestReceivedCall) DoAndReturn(f func(proto
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateSendWindow mocks base method.
|
// UpdateSendWindow mocks base method.
|
||||||
func (m *MockStreamFlowController) UpdateSendWindow(arg0 protocol.ByteCount) {
|
func (m *MockStreamFlowController) UpdateSendWindow(arg0 protocol.ByteCount) bool {
|
||||||
m.ctrl.T.Helper()
|
m.ctrl.T.Helper()
|
||||||
m.ctrl.Call(m, "UpdateSendWindow", arg0)
|
ret := m.ctrl.Call(m, "UpdateSendWindow", arg0)
|
||||||
|
ret0, _ := ret[0].(bool)
|
||||||
|
return ret0
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateSendWindow indicates an expected call of UpdateSendWindow.
|
// UpdateSendWindow indicates an expected call of UpdateSendWindow.
|
||||||
|
@ -318,19 +320,19 @@ type StreamFlowControllerUpdateSendWindowCall struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return rewrite *gomock.Call.Return
|
// Return rewrite *gomock.Call.Return
|
||||||
func (c *StreamFlowControllerUpdateSendWindowCall) Return() *StreamFlowControllerUpdateSendWindowCall {
|
func (c *StreamFlowControllerUpdateSendWindowCall) Return(arg0 bool) *StreamFlowControllerUpdateSendWindowCall {
|
||||||
c.Call = c.Call.Return()
|
c.Call = c.Call.Return(arg0)
|
||||||
return c
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
// Do rewrite *gomock.Call.Do
|
// Do rewrite *gomock.Call.Do
|
||||||
func (c *StreamFlowControllerUpdateSendWindowCall) Do(f func(protocol.ByteCount)) *StreamFlowControllerUpdateSendWindowCall {
|
func (c *StreamFlowControllerUpdateSendWindowCall) Do(f func(protocol.ByteCount) bool) *StreamFlowControllerUpdateSendWindowCall {
|
||||||
c.Call = c.Call.Do(f)
|
c.Call = c.Call.Do(f)
|
||||||
return c
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
// DoAndReturn rewrite *gomock.Call.DoAndReturn
|
// DoAndReturn rewrite *gomock.Call.DoAndReturn
|
||||||
func (c *StreamFlowControllerUpdateSendWindowCall) DoAndReturn(f func(protocol.ByteCount)) *StreamFlowControllerUpdateSendWindowCall {
|
func (c *StreamFlowControllerUpdateSendWindowCall) DoAndReturn(f func(protocol.ByteCount) bool) *StreamFlowControllerUpdateSendWindowCall {
|
||||||
c.Call = c.Call.DoAndReturn(f)
|
c.Call = c.Call.DoAndReturn(f)
|
||||||
return c
|
return c
|
||||||
}
|
}
|
||||||
|
|
|
@ -404,11 +404,13 @@ func (s *sendStream) cancelWriteImpl(errorCode qerr.StreamErrorCode, remote bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *sendStream) updateSendWindow(limit protocol.ByteCount) {
|
func (s *sendStream) updateSendWindow(limit protocol.ByteCount) {
|
||||||
|
updated := s.flowController.UpdateSendWindow(limit)
|
||||||
|
if !updated { // duplicate or reordered MAX_STREAM_DATA frame
|
||||||
|
return
|
||||||
|
}
|
||||||
s.mutex.Lock()
|
s.mutex.Lock()
|
||||||
hasStreamData := s.dataForWriting != nil || s.nextFrame != nil
|
hasStreamData := s.dataForWriting != nil || s.nextFrame != nil
|
||||||
s.mutex.Unlock()
|
s.mutex.Unlock()
|
||||||
|
|
||||||
s.flowController.UpdateSendWindow(limit)
|
|
||||||
if hasStreamData {
|
if hasStreamData {
|
||||||
s.sender.onHasStreamData(s.streamID)
|
s.sender.onHasStreamData(s.streamID)
|
||||||
}
|
}
|
||||||
|
|
|
@ -682,7 +682,7 @@ var _ = Describe("Send Stream", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("says when it has data for sending", func() {
|
It("says when it has data for sending", func() {
|
||||||
mockFC.EXPECT().UpdateSendWindow(gomock.Any())
|
mockFC.EXPECT().UpdateSendWindow(gomock.Any()).Return(true)
|
||||||
mockSender.EXPECT().onHasStreamData(streamID)
|
mockSender.EXPECT().onHasStreamData(streamID)
|
||||||
done := make(chan struct{})
|
done := make(chan struct{})
|
||||||
go func() {
|
go func() {
|
||||||
|
@ -698,6 +698,24 @@ var _ = Describe("Send Stream", func() {
|
||||||
str.closeForShutdown(nil)
|
str.closeForShutdown(nil)
|
||||||
Eventually(done).Should(BeClosed())
|
Eventually(done).Should(BeClosed())
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("doesn't say it has data for sending if the MAX_STREAM_DATA frame was reordered", func() {
|
||||||
|
mockFC.EXPECT().UpdateSendWindow(gomock.Any()).Return(false) // reordered frame
|
||||||
|
mockSender.EXPECT().onHasStreamData(streamID)
|
||||||
|
done := make(chan struct{})
|
||||||
|
go func() {
|
||||||
|
defer GinkgoRecover()
|
||||||
|
_, err := str.Write([]byte("foobar"))
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
close(done)
|
||||||
|
}()
|
||||||
|
waitForWrite()
|
||||||
|
// don't expect any calls to onHasStreamData
|
||||||
|
str.updateSendWindow(42)
|
||||||
|
// make sure the Write go routine returns
|
||||||
|
str.closeForShutdown(nil)
|
||||||
|
Eventually(done).Should(BeClosed())
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
Context("stream cancellations", func() {
|
Context("stream cancellations", func() {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue