mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-05 21:27:35 +03:00
update the flow control windows of streams opened in 0-RTT
The server might have increased the initial flow control window. We need to make sure to inform all streams opened during during the 0-RTT period.
This commit is contained in:
parent
c27f5a55b3
commit
7cd4dea764
8 changed files with 85 additions and 2 deletions
|
@ -278,7 +278,9 @@ func (m *streamsMap) HandleMaxStreamsFrame(f *wire.MaxStreamsFrame) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (m *streamsMap) UpdateLimits(p *wire.TransportParameters) {
|
func (m *streamsMap) UpdateLimits(p *wire.TransportParameters) {
|
||||||
|
m.outgoingBidiStreams.UpdateSendWindow(p.InitialMaxStreamDataBidiRemote)
|
||||||
m.outgoingBidiStreams.SetMaxStream(p.MaxBidiStreamNum)
|
m.outgoingBidiStreams.SetMaxStream(p.MaxBidiStreamNum)
|
||||||
|
m.outgoingUniStreams.UpdateSendWindow(p.InitialMaxStreamDataUni)
|
||||||
m.outgoingUniStreams.SetMaxStream(p.MaxUniStreamNum)
|
m.outgoingUniStreams.SetMaxStream(p.MaxUniStreamNum)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -11,6 +11,7 @@ import (
|
||||||
// This definition must be in a file that Genny doesn't process.
|
// This definition must be in a file that Genny doesn't process.
|
||||||
type item interface {
|
type item interface {
|
||||||
generic.Type
|
generic.Type
|
||||||
|
updateSendWindow(protocol.ByteCount)
|
||||||
closeForShutdown(error)
|
closeForShutdown(error)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -20,6 +20,7 @@ type mockGenericStream struct {
|
||||||
|
|
||||||
closed bool
|
closed bool
|
||||||
closeErr error
|
closeErr error
|
||||||
|
sendWindow protocol.ByteCount
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *mockGenericStream) closeForShutdown(err error) {
|
func (s *mockGenericStream) closeForShutdown(err error) {
|
||||||
|
@ -27,6 +28,10 @@ func (s *mockGenericStream) closeForShutdown(err error) {
|
||||||
s.closeErr = err
|
s.closeErr = err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *mockGenericStream) updateSendWindow(limit protocol.ByteCount) {
|
||||||
|
s.sendWindow = limit
|
||||||
|
}
|
||||||
|
|
||||||
var _ = Describe("Streams Map (incoming)", func() {
|
var _ = Describe("Streams Map (incoming)", func() {
|
||||||
var (
|
var (
|
||||||
m *incomingItemsMap
|
m *incomingItemsMap
|
||||||
|
|
|
@ -180,6 +180,17 @@ func (m *outgoingBidiStreamsMap) SetMaxStream(num protocol.StreamNum) {
|
||||||
m.unblockOpenSync()
|
m.unblockOpenSync()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UpdateSendWindow is called when the peer's transport parameters are received.
|
||||||
|
// Only in the case of a 0-RTT handshake will we have open streams at this point.
|
||||||
|
// We might need to update the send window, in case the server increased it.
|
||||||
|
func (m *outgoingBidiStreamsMap) UpdateSendWindow(limit protocol.ByteCount) {
|
||||||
|
m.mutex.Lock()
|
||||||
|
for _, str := range m.streams {
|
||||||
|
str.updateSendWindow(limit)
|
||||||
|
}
|
||||||
|
m.mutex.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
// unblockOpenSync unblocks the next OpenStreamSync go-routine to open a new stream
|
// unblockOpenSync unblocks the next OpenStreamSync go-routine to open a new stream
|
||||||
func (m *outgoingBidiStreamsMap) unblockOpenSync() {
|
func (m *outgoingBidiStreamsMap) unblockOpenSync() {
|
||||||
if len(m.openQueue) == 0 {
|
if len(m.openQueue) == 0 {
|
||||||
|
|
|
@ -178,6 +178,17 @@ func (m *outgoingItemsMap) SetMaxStream(num protocol.StreamNum) {
|
||||||
m.unblockOpenSync()
|
m.unblockOpenSync()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UpdateSendWindow is called when the peer's transport parameters are received.
|
||||||
|
// Only in the case of a 0-RTT handshake will we have open streams at this point.
|
||||||
|
// We might need to update the send window, in case the server increased it.
|
||||||
|
func (m *outgoingItemsMap) UpdateSendWindow(limit protocol.ByteCount) {
|
||||||
|
m.mutex.Lock()
|
||||||
|
for _, str := range m.streams {
|
||||||
|
str.updateSendWindow(limit)
|
||||||
|
}
|
||||||
|
m.mutex.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
// unblockOpenSync unblocks the next OpenStreamSync go-routine to open a new stream
|
// unblockOpenSync unblocks the next OpenStreamSync go-routine to open a new stream
|
||||||
func (m *outgoingItemsMap) unblockOpenSync() {
|
func (m *outgoingItemsMap) unblockOpenSync() {
|
||||||
if len(m.openQueue) == 0 {
|
if len(m.openQueue) == 0 {
|
||||||
|
|
|
@ -112,6 +112,16 @@ var _ = Describe("Streams Map (outgoing)", func() {
|
||||||
Expect(str2.(*mockGenericStream).closed).To(BeTrue())
|
Expect(str2.(*mockGenericStream).closed).To(BeTrue())
|
||||||
Expect(str2.(*mockGenericStream).closeErr).To(MatchError(testErr))
|
Expect(str2.(*mockGenericStream).closeErr).To(MatchError(testErr))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("updates the send window", func() {
|
||||||
|
str1, err := m.OpenStream()
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
str2, err := m.OpenStream()
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
m.UpdateSendWindow(1337)
|
||||||
|
Expect(str1.(*mockGenericStream).sendWindow).To(BeEquivalentTo(1337))
|
||||||
|
Expect(str2.(*mockGenericStream).sendWindow).To(BeEquivalentTo(1337))
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
Context("with stream ID limits", func() {
|
Context("with stream ID limits", func() {
|
||||||
|
|
|
@ -180,6 +180,17 @@ func (m *outgoingUniStreamsMap) SetMaxStream(num protocol.StreamNum) {
|
||||||
m.unblockOpenSync()
|
m.unblockOpenSync()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// UpdateSendWindow is called when the peer's transport parameters are received.
|
||||||
|
// Only in the case of a 0-RTT handshake will we have open streams at this point.
|
||||||
|
// We might need to update the send window, in case the server increased it.
|
||||||
|
func (m *outgoingUniStreamsMap) UpdateSendWindow(limit protocol.ByteCount) {
|
||||||
|
m.mutex.Lock()
|
||||||
|
for _, str := range m.streams {
|
||||||
|
str.updateSendWindow(limit)
|
||||||
|
}
|
||||||
|
m.mutex.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
// unblockOpenSync unblocks the next OpenStreamSync go-routine to open a new stream
|
// unblockOpenSync unblocks the next OpenStreamSync go-routine to open a new stream
|
||||||
func (m *outgoingUniStreamsMap) unblockOpenSync() {
|
func (m *outgoingUniStreamsMap) unblockOpenSync() {
|
||||||
if len(m.openQueue) == 0 {
|
if len(m.openQueue) == 0 {
|
||||||
|
|
|
@ -348,6 +348,38 @@ var _ = Describe("Streams Map", func() {
|
||||||
expectTooManyStreamsError(err)
|
expectTooManyStreamsError(err)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
if perspective == protocol.PerspectiveClient {
|
||||||
|
It("applies parameters to existing streams (needed for 0-RTT)", func() {
|
||||||
|
m.UpdateLimits(&wire.TransportParameters{
|
||||||
|
MaxBidiStreamNum: 1000,
|
||||||
|
MaxUniStreamNum: 1000,
|
||||||
|
})
|
||||||
|
flowControllers := make(map[protocol.StreamID]*mocks.MockStreamFlowController)
|
||||||
|
m.newFlowController = func(id protocol.StreamID) flowcontrol.StreamFlowController {
|
||||||
|
fc := mocks.NewMockStreamFlowController(mockCtrl)
|
||||||
|
flowControllers[id] = fc
|
||||||
|
return fc
|
||||||
|
}
|
||||||
|
|
||||||
|
str, err := m.OpenStream()
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
unistr, err := m.OpenUniStream()
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
|
||||||
|
Expect(flowControllers).To(HaveKey(str.StreamID()))
|
||||||
|
flowControllers[str.StreamID()].EXPECT().UpdateSendWindow(protocol.ByteCount(4321))
|
||||||
|
Expect(flowControllers).To(HaveKey(unistr.StreamID()))
|
||||||
|
flowControllers[unistr.StreamID()].EXPECT().UpdateSendWindow(protocol.ByteCount(1234))
|
||||||
|
|
||||||
|
m.UpdateLimits(&wire.TransportParameters{
|
||||||
|
MaxBidiStreamNum: 1000,
|
||||||
|
InitialMaxStreamDataUni: 1234,
|
||||||
|
MaxUniStreamNum: 1000,
|
||||||
|
InitialMaxStreamDataBidiRemote: 4321,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
Context("handling MAX_STREAMS frames", func() {
|
Context("handling MAX_STREAMS frames", func() {
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
mockSender.EXPECT().queueControlFrame(gomock.Any()).AnyTimes()
|
mockSender.EXPECT().queueControlFrame(gomock.Any()).AnyTimes()
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue