mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-06 05:37:36 +03:00
170 lines
4.7 KiB
Go
170 lines
4.7 KiB
Go
package flowcontrol
|
|
|
|
import (
|
|
"sync"
|
|
|
|
"github.com/lucas-clemente/quic-go/handshake"
|
|
"github.com/lucas-clemente/quic-go/protocol"
|
|
)
|
|
|
|
type flowController struct {
|
|
streamID protocol.StreamID
|
|
|
|
connectionParametersManager *handshake.ConnectionParametersManager
|
|
|
|
bytesSent protocol.ByteCount
|
|
sendFlowControlWindow protocol.ByteCount
|
|
lastBlockedSentForOffset protocol.ByteCount
|
|
|
|
bytesRead protocol.ByteCount
|
|
highestReceived protocol.ByteCount
|
|
receiveFlowControlWindow protocol.ByteCount
|
|
receiveFlowControlWindowIncrement protocol.ByteCount
|
|
|
|
mutex sync.RWMutex
|
|
}
|
|
|
|
// NewFlowController gets a new flow controller
|
|
func NewFlowController(streamID protocol.StreamID, connectionParametersManager *handshake.ConnectionParametersManager) FlowController {
|
|
fc := flowController{
|
|
streamID: streamID,
|
|
connectionParametersManager: connectionParametersManager,
|
|
}
|
|
|
|
if streamID == 0 {
|
|
fc.receiveFlowControlWindow = connectionParametersManager.GetReceiveConnectionFlowControlWindow()
|
|
fc.receiveFlowControlWindowIncrement = fc.receiveFlowControlWindow
|
|
} else {
|
|
fc.receiveFlowControlWindow = connectionParametersManager.GetReceiveStreamFlowControlWindow()
|
|
fc.receiveFlowControlWindowIncrement = fc.receiveFlowControlWindow
|
|
}
|
|
|
|
return &fc
|
|
}
|
|
|
|
func (c *flowController) getSendFlowControlWindow() protocol.ByteCount {
|
|
c.mutex.RLock()
|
|
defer c.mutex.RUnlock()
|
|
|
|
if c.sendFlowControlWindow == 0 {
|
|
if c.streamID == 0 {
|
|
return c.connectionParametersManager.GetSendConnectionFlowControlWindow()
|
|
}
|
|
return c.connectionParametersManager.GetSendStreamFlowControlWindow()
|
|
}
|
|
return c.sendFlowControlWindow
|
|
}
|
|
|
|
func (c *flowController) AddBytesSent(n protocol.ByteCount) {
|
|
c.mutex.Lock()
|
|
defer c.mutex.Unlock()
|
|
|
|
c.bytesSent += n
|
|
}
|
|
|
|
// UpdateSendWindow should be called after receiving a WindowUpdateFrame
|
|
// it returns true if the window was actually updated
|
|
func (c *flowController) UpdateSendWindow(newOffset protocol.ByteCount) bool {
|
|
c.mutex.Lock()
|
|
defer c.mutex.Unlock()
|
|
|
|
if newOffset > c.sendFlowControlWindow {
|
|
c.sendFlowControlWindow = newOffset
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (c *flowController) SendWindowSize() protocol.ByteCount {
|
|
c.mutex.RLock()
|
|
defer c.mutex.RUnlock()
|
|
|
|
sendFlowControlWindow := c.getSendFlowControlWindow()
|
|
|
|
if c.bytesSent > sendFlowControlWindow { // should never happen, but make sure we don't do an underflow here
|
|
return 0
|
|
}
|
|
return sendFlowControlWindow - c.bytesSent
|
|
}
|
|
|
|
// UpdateHighestReceived updates the highestReceived value, if the byteOffset is higher
|
|
// Should **only** be used for the stream-level FlowController
|
|
func (c *flowController) UpdateHighestReceived(byteOffset protocol.ByteCount) protocol.ByteCount {
|
|
c.mutex.Lock()
|
|
defer c.mutex.Unlock()
|
|
|
|
if byteOffset > c.highestReceived {
|
|
increment := byteOffset - c.highestReceived
|
|
c.highestReceived = byteOffset
|
|
return increment
|
|
}
|
|
return 0
|
|
}
|
|
|
|
// IncrementHighestReceived adds an increment to the highestReceived value
|
|
// Should **only** be used for the connection-level FlowController
|
|
func (c *flowController) IncrementHighestReceived(increment protocol.ByteCount) {
|
|
c.mutex.Lock()
|
|
defer c.mutex.Unlock()
|
|
|
|
c.highestReceived += increment
|
|
}
|
|
|
|
func (c *flowController) AddBytesRead(n protocol.ByteCount) {
|
|
c.mutex.Lock()
|
|
defer c.mutex.Unlock()
|
|
|
|
c.bytesRead += n
|
|
}
|
|
|
|
// MaybeTriggerBlocked determines if it is necessary to send a Blocked for this stream
|
|
// it makes sure that only one Blocked is sent for each offset
|
|
func (c *flowController) MaybeTriggerBlocked() bool {
|
|
if c.SendWindowSize() != 0 {
|
|
return false
|
|
}
|
|
|
|
sendFlowControlWindow := c.getSendFlowControlWindow()
|
|
|
|
c.mutex.Lock()
|
|
defer c.mutex.Unlock()
|
|
|
|
if c.lastBlockedSentForOffset == sendFlowControlWindow {
|
|
return false
|
|
}
|
|
|
|
c.lastBlockedSentForOffset = sendFlowControlWindow
|
|
return true
|
|
}
|
|
|
|
// MaybeTriggerWindowUpdate determines if it is necessary to send a WindowUpdate
|
|
// if so, it returns true and the offset of the window
|
|
func (c *flowController) MaybeTriggerWindowUpdate() (bool, protocol.ByteCount) {
|
|
c.mutex.Lock()
|
|
defer c.mutex.Unlock()
|
|
|
|
diff := c.receiveFlowControlWindow - c.bytesRead
|
|
// Chromium implements the same threshold
|
|
if diff < (c.receiveFlowControlWindowIncrement / 2) {
|
|
c.receiveFlowControlWindow += c.receiveFlowControlWindowIncrement
|
|
return true, c.bytesRead + c.receiveFlowControlWindowIncrement
|
|
}
|
|
return false, 0
|
|
}
|
|
|
|
func (c *flowController) CheckFlowControlViolation() bool {
|
|
c.mutex.Lock()
|
|
defer c.mutex.Unlock()
|
|
|
|
if c.highestReceived > c.receiveFlowControlWindow {
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (c *flowController) GetHighestReceived() protocol.ByteCount {
|
|
c.mutex.RLock()
|
|
defer c.mutex.RUnlock()
|
|
|
|
return c.highestReceived
|
|
}
|