mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-05 13:17:36 +03:00
add a callback to block window increases to the connection flow controller
This commit is contained in:
parent
a98e60c28c
commit
f9904c7c45
5 changed files with 75 additions and 8 deletions
|
@ -23,6 +23,8 @@ type baseFlowController struct {
|
||||||
receiveWindowSize protocol.ByteCount
|
receiveWindowSize protocol.ByteCount
|
||||||
maxReceiveWindowSize protocol.ByteCount
|
maxReceiveWindowSize protocol.ByteCount
|
||||||
|
|
||||||
|
allowWindowIncrease func(size protocol.ByteCount) bool
|
||||||
|
|
||||||
epochStartTime time.Time
|
epochStartTime time.Time
|
||||||
epochStartOffset protocol.ByteCount
|
epochStartOffset protocol.ByteCount
|
||||||
rttStats *utils.RTTStats
|
rttStats *utils.RTTStats
|
||||||
|
@ -105,7 +107,10 @@ func (c *baseFlowController) maybeAdjustWindowSize() {
|
||||||
now := time.Now()
|
now := time.Now()
|
||||||
if now.Sub(c.epochStartTime) < time.Duration(4*fraction*float64(rtt)) {
|
if now.Sub(c.epochStartTime) < time.Duration(4*fraction*float64(rtt)) {
|
||||||
// window is consumed too fast, try to increase the window size
|
// window is consumed too fast, try to increase the window size
|
||||||
c.receiveWindowSize = utils.MinByteCount(2*c.receiveWindowSize, c.maxReceiveWindowSize)
|
newSize := utils.MinByteCount(2*c.receiveWindowSize, c.maxReceiveWindowSize)
|
||||||
|
if newSize > c.receiveWindowSize && (c.allowWindowIncrease == nil || c.allowWindowIncrease(newSize-c.receiveWindowSize)) {
|
||||||
|
c.receiveWindowSize = newSize
|
||||||
|
}
|
||||||
}
|
}
|
||||||
c.startNewAutoTuningEpoch(now)
|
c.startNewAutoTuningEpoch(now)
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,6 +24,7 @@ func NewConnectionFlowController(
|
||||||
receiveWindow protocol.ByteCount,
|
receiveWindow protocol.ByteCount,
|
||||||
maxReceiveWindow protocol.ByteCount,
|
maxReceiveWindow protocol.ByteCount,
|
||||||
queueWindowUpdate func(),
|
queueWindowUpdate func(),
|
||||||
|
allowWindowIncrease func(size protocol.ByteCount) bool,
|
||||||
rttStats *utils.RTTStats,
|
rttStats *utils.RTTStats,
|
||||||
logger utils.Logger,
|
logger utils.Logger,
|
||||||
) ConnectionFlowController {
|
) ConnectionFlowController {
|
||||||
|
@ -33,6 +34,7 @@ func NewConnectionFlowController(
|
||||||
receiveWindow: receiveWindow,
|
receiveWindow: receiveWindow,
|
||||||
receiveWindowSize: receiveWindow,
|
receiveWindowSize: receiveWindow,
|
||||||
maxReceiveWindowSize: maxReceiveWindow,
|
maxReceiveWindowSize: maxReceiveWindow,
|
||||||
|
allowWindowIncrease: allowWindowIncrease,
|
||||||
logger: logger,
|
logger: logger,
|
||||||
},
|
},
|
||||||
queueWindowUpdate: queueWindowUpdate,
|
queueWindowUpdate: queueWindowUpdate,
|
||||||
|
@ -85,13 +87,16 @@ func (c *connectionFlowController) EnsureMinimumWindowSize(inc protocol.ByteCoun
|
||||||
c.mutex.Lock()
|
c.mutex.Lock()
|
||||||
if inc > c.receiveWindowSize {
|
if inc > c.receiveWindowSize {
|
||||||
c.logger.Debugf("Increasing receive flow control window for the connection to %d kB, in response to stream flow control window increase", c.receiveWindowSize/(1<<10))
|
c.logger.Debugf("Increasing receive flow control window for the connection to %d kB, in response to stream flow control window increase", c.receiveWindowSize/(1<<10))
|
||||||
c.receiveWindowSize = utils.MinByteCount(inc, c.maxReceiveWindowSize)
|
newSize := utils.MinByteCount(inc, c.maxReceiveWindowSize)
|
||||||
|
if delta := newSize - c.receiveWindowSize; delta > 0 && c.allowWindowIncrease(delta) {
|
||||||
|
c.receiveWindowSize = newSize
|
||||||
|
}
|
||||||
c.startNewAutoTuningEpoch(time.Now())
|
c.startNewAutoTuningEpoch(time.Now())
|
||||||
}
|
}
|
||||||
c.mutex.Unlock()
|
c.mutex.Unlock()
|
||||||
}
|
}
|
||||||
|
|
||||||
// The flow controller is reset when 0-RTT is rejected.
|
// Reset rests the flow controller. This happens when 0-RTT is rejected.
|
||||||
// All stream data is invalidated, it's if we had never opened a stream and never sent any data.
|
// All stream data is invalidated, it's if we had never opened a stream and never sent any data.
|
||||||
// At that point, we only have sent stream data, but we didn't have the keys to open 1-RTT keys yet.
|
// At that point, we only have sent stream data, but we didn't have the keys to open 1-RTT keys yet.
|
||||||
func (c *connectionFlowController) Reset() error {
|
func (c *connectionFlowController) Reset() error {
|
||||||
|
|
|
@ -27,6 +27,7 @@ var _ = Describe("Connection Flow controller", func() {
|
||||||
controller.rttStats = &utils.RTTStats{}
|
controller.rttStats = &utils.RTTStats{}
|
||||||
controller.logger = utils.DefaultLogger
|
controller.logger = utils.DefaultLogger
|
||||||
controller.queueWindowUpdate = func() { queuedWindowUpdate = true }
|
controller.queueWindowUpdate = func() { queuedWindowUpdate = true }
|
||||||
|
controller.allowWindowIncrease = func(protocol.ByteCount) bool { return true }
|
||||||
})
|
})
|
||||||
|
|
||||||
Context("Constructor", func() {
|
Context("Constructor", func() {
|
||||||
|
@ -36,7 +37,13 @@ var _ = Describe("Connection Flow controller", func() {
|
||||||
receiveWindow := protocol.ByteCount(2000)
|
receiveWindow := protocol.ByteCount(2000)
|
||||||
maxReceiveWindow := protocol.ByteCount(3000)
|
maxReceiveWindow := protocol.ByteCount(3000)
|
||||||
|
|
||||||
fc := NewConnectionFlowController(receiveWindow, maxReceiveWindow, nil, rttStats, utils.DefaultLogger).(*connectionFlowController)
|
fc := NewConnectionFlowController(
|
||||||
|
receiveWindow,
|
||||||
|
maxReceiveWindow,
|
||||||
|
nil,
|
||||||
|
func(protocol.ByteCount) bool { return true },
|
||||||
|
rttStats,
|
||||||
|
utils.DefaultLogger).(*connectionFlowController)
|
||||||
Expect(fc.receiveWindow).To(Equal(receiveWindow))
|
Expect(fc.receiveWindow).To(Equal(receiveWindow))
|
||||||
Expect(fc.maxReceiveWindowSize).To(Equal(maxReceiveWindow))
|
Expect(fc.maxReceiveWindowSize).To(Equal(maxReceiveWindow))
|
||||||
})
|
})
|
||||||
|
@ -78,6 +85,11 @@ var _ = Describe("Connection Flow controller", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("auto-tunes the window", func() {
|
It("auto-tunes the window", func() {
|
||||||
|
var allowed protocol.ByteCount
|
||||||
|
controller.allowWindowIncrease = func(size protocol.ByteCount) bool {
|
||||||
|
allowed = size
|
||||||
|
return true
|
||||||
|
}
|
||||||
oldOffset := controller.bytesRead
|
oldOffset := controller.bytesRead
|
||||||
oldWindowSize := controller.receiveWindowSize
|
oldWindowSize := controller.receiveWindowSize
|
||||||
rtt := scaleDuration(20 * time.Millisecond)
|
rtt := scaleDuration(20 * time.Millisecond)
|
||||||
|
@ -90,6 +102,23 @@ var _ = Describe("Connection Flow controller", func() {
|
||||||
newWindowSize := controller.receiveWindowSize
|
newWindowSize := controller.receiveWindowSize
|
||||||
Expect(newWindowSize).To(Equal(2 * oldWindowSize))
|
Expect(newWindowSize).To(Equal(2 * oldWindowSize))
|
||||||
Expect(offset).To(Equal(oldOffset + dataRead + newWindowSize))
|
Expect(offset).To(Equal(oldOffset + dataRead + newWindowSize))
|
||||||
|
Expect(allowed).To(Equal(oldWindowSize))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("doesn't auto-tune the window if it's not allowed", func() {
|
||||||
|
controller.allowWindowIncrease = func(protocol.ByteCount) bool { return false }
|
||||||
|
oldOffset := controller.bytesRead
|
||||||
|
oldWindowSize := controller.receiveWindowSize
|
||||||
|
rtt := scaleDuration(20 * time.Millisecond)
|
||||||
|
setRtt(rtt)
|
||||||
|
controller.epochStartTime = time.Now().Add(-time.Millisecond)
|
||||||
|
controller.epochStartOffset = oldOffset
|
||||||
|
dataRead := oldWindowSize/2 + 1
|
||||||
|
controller.AddBytesRead(dataRead)
|
||||||
|
offset := controller.GetWindowUpdate()
|
||||||
|
newWindowSize := controller.receiveWindowSize
|
||||||
|
Expect(newWindowSize).To(Equal(oldWindowSize))
|
||||||
|
Expect(offset).To(Equal(oldOffset + dataRead + newWindowSize))
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
|
@ -21,8 +21,15 @@ var _ = Describe("Stream Flow controller", func() {
|
||||||
queuedWindowUpdate = false
|
queuedWindowUpdate = false
|
||||||
rttStats := &utils.RTTStats{}
|
rttStats := &utils.RTTStats{}
|
||||||
controller = &streamFlowController{
|
controller = &streamFlowController{
|
||||||
streamID: 10,
|
streamID: 10,
|
||||||
connection: NewConnectionFlowController(1000, 1000, func() {}, rttStats, utils.DefaultLogger).(*connectionFlowController),
|
connection: NewConnectionFlowController(
|
||||||
|
1000,
|
||||||
|
1000,
|
||||||
|
func() {},
|
||||||
|
func(protocol.ByteCount) bool { return true },
|
||||||
|
rttStats,
|
||||||
|
utils.DefaultLogger,
|
||||||
|
).(*connectionFlowController),
|
||||||
}
|
}
|
||||||
controller.maxReceiveWindowSize = 10000
|
controller.maxReceiveWindowSize = 10000
|
||||||
controller.rttStats = rttStats
|
controller.rttStats = rttStats
|
||||||
|
@ -37,7 +44,7 @@ var _ = Describe("Stream Flow controller", func() {
|
||||||
const sendWindow protocol.ByteCount = 4000
|
const sendWindow protocol.ByteCount = 4000
|
||||||
|
|
||||||
It("sets the send and receive windows", func() {
|
It("sets the send and receive windows", func() {
|
||||||
cc := NewConnectionFlowController(0, 0, nil, nil, utils.DefaultLogger)
|
cc := NewConnectionFlowController(0, 0, nil, func(protocol.ByteCount) bool { return true }, nil, utils.DefaultLogger)
|
||||||
fc := NewStreamFlowController(5, cc, receiveWindow, maxReceiveWindow, sendWindow, nil, rttStats, utils.DefaultLogger).(*streamFlowController)
|
fc := NewStreamFlowController(5, cc, receiveWindow, maxReceiveWindow, sendWindow, nil, rttStats, utils.DefaultLogger).(*streamFlowController)
|
||||||
Expect(fc.streamID).To(Equal(protocol.StreamID(5)))
|
Expect(fc.streamID).To(Equal(protocol.StreamID(5)))
|
||||||
Expect(fc.receiveWindow).To(Equal(receiveWindow))
|
Expect(fc.receiveWindow).To(Equal(receiveWindow))
|
||||||
|
@ -52,7 +59,7 @@ var _ = Describe("Stream Flow controller", func() {
|
||||||
queued = true
|
queued = true
|
||||||
}
|
}
|
||||||
|
|
||||||
cc := NewConnectionFlowController(receiveWindow, maxReceiveWindow, func() {}, nil, utils.DefaultLogger)
|
cc := NewConnectionFlowController(receiveWindow, maxReceiveWindow, func() {}, func(protocol.ByteCount) bool { return true }, nil, utils.DefaultLogger)
|
||||||
fc := NewStreamFlowController(5, cc, receiveWindow, maxReceiveWindow, sendWindow, queueWindowUpdate, rttStats, utils.DefaultLogger).(*streamFlowController)
|
fc := NewStreamFlowController(5, cc, receiveWindow, maxReceiveWindow, sendWindow, queueWindowUpdate, rttStats, utils.DefaultLogger).(*streamFlowController)
|
||||||
fc.AddBytesRead(receiveWindow)
|
fc.AddBytesRead(receiveWindow)
|
||||||
Expect(queued).To(BeTrue())
|
Expect(queued).To(BeTrue())
|
||||||
|
@ -190,6 +197,11 @@ var _ = Describe("Stream Flow controller", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("tells the connection flow controller when the window was auto-tuned", func() {
|
It("tells the connection flow controller when the window was auto-tuned", func() {
|
||||||
|
var allowed protocol.ByteCount
|
||||||
|
controller.connection.(*connectionFlowController).allowWindowIncrease = func(size protocol.ByteCount) bool {
|
||||||
|
allowed = size
|
||||||
|
return true
|
||||||
|
}
|
||||||
oldOffset := controller.bytesRead
|
oldOffset := controller.bytesRead
|
||||||
setRtt(scaleDuration(20 * time.Millisecond))
|
setRtt(scaleDuration(20 * time.Millisecond))
|
||||||
controller.epochStartOffset = oldOffset
|
controller.epochStartOffset = oldOffset
|
||||||
|
@ -198,9 +210,24 @@ var _ = Describe("Stream Flow controller", func() {
|
||||||
offset := controller.GetWindowUpdate()
|
offset := controller.GetWindowUpdate()
|
||||||
Expect(offset).To(Equal(oldOffset + 55 + 2*oldWindowSize))
|
Expect(offset).To(Equal(oldOffset + 55 + 2*oldWindowSize))
|
||||||
Expect(controller.receiveWindowSize).To(Equal(2 * oldWindowSize))
|
Expect(controller.receiveWindowSize).To(Equal(2 * oldWindowSize))
|
||||||
|
Expect(allowed).To(Equal(oldWindowSize))
|
||||||
Expect(controller.connection.(*connectionFlowController).receiveWindowSize).To(Equal(protocol.ByteCount(float64(controller.receiveWindowSize) * protocol.ConnectionFlowControlMultiplier)))
|
Expect(controller.connection.(*connectionFlowController).receiveWindowSize).To(Equal(protocol.ByteCount(float64(controller.receiveWindowSize) * protocol.ConnectionFlowControlMultiplier)))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("doesn't increase the connection flow control window if it's not allowed", func() {
|
||||||
|
oldOffset := controller.bytesRead
|
||||||
|
oldConnectionSize := controller.connection.(*connectionFlowController).receiveWindowSize
|
||||||
|
controller.connection.(*connectionFlowController).allowWindowIncrease = func(protocol.ByteCount) bool { return false }
|
||||||
|
setRtt(scaleDuration(20 * time.Millisecond))
|
||||||
|
controller.epochStartOffset = oldOffset
|
||||||
|
controller.epochStartTime = time.Now().Add(-time.Millisecond)
|
||||||
|
controller.AddBytesRead(55)
|
||||||
|
offset := controller.GetWindowUpdate()
|
||||||
|
Expect(offset).To(Equal(oldOffset + 55 + 2*oldWindowSize))
|
||||||
|
Expect(controller.receiveWindowSize).To(Equal(2 * oldWindowSize))
|
||||||
|
Expect(controller.connection.(*connectionFlowController).receiveWindowSize).To(Equal(oldConnectionSize))
|
||||||
|
})
|
||||||
|
|
||||||
It("sends a connection-level window update when a large stream is abandoned", func() {
|
It("sends a connection-level window update when a large stream is abandoned", func() {
|
||||||
Expect(controller.UpdateHighestReceived(90, true)).To(Succeed())
|
Expect(controller.UpdateHighestReceived(90, true)).To(Succeed())
|
||||||
Expect(controller.connection.GetWindowUpdate()).To(BeZero())
|
Expect(controller.connection.GetWindowUpdate()).To(BeZero())
|
||||||
|
|
|
@ -506,6 +506,7 @@ func (s *session) preSetup() {
|
||||||
protocol.ByteCount(s.config.InitialConnectionReceiveWindow),
|
protocol.ByteCount(s.config.InitialConnectionReceiveWindow),
|
||||||
protocol.ByteCount(s.config.MaxConnectionReceiveWindow),
|
protocol.ByteCount(s.config.MaxConnectionReceiveWindow),
|
||||||
s.onHasConnectionWindowUpdate,
|
s.onHasConnectionWindowUpdate,
|
||||||
|
func(protocol.ByteCount) bool { return true },
|
||||||
s.rttStats,
|
s.rttStats,
|
||||||
s.logger,
|
s.logger,
|
||||||
)
|
)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue