mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
uTLS is not yet bumped to the new version, so this commit breaks the dependencies relationship by getting rid of the local replace.
149 lines
4.7 KiB
Go
149 lines
4.7 KiB
Go
package flowcontrol
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/refraction-networking/uquic/internal/protocol"
|
|
"github.com/refraction-networking/uquic/internal/qerr"
|
|
"github.com/refraction-networking/uquic/internal/utils"
|
|
)
|
|
|
|
type streamFlowController struct {
|
|
baseFlowController
|
|
|
|
streamID protocol.StreamID
|
|
|
|
queueWindowUpdate func()
|
|
|
|
connection connectionFlowControllerI
|
|
|
|
receivedFinalOffset bool
|
|
}
|
|
|
|
var _ StreamFlowController = &streamFlowController{}
|
|
|
|
// NewStreamFlowController gets a new flow controller for a stream
|
|
func NewStreamFlowController(
|
|
streamID protocol.StreamID,
|
|
cfc ConnectionFlowController,
|
|
receiveWindow protocol.ByteCount,
|
|
maxReceiveWindow protocol.ByteCount,
|
|
initialSendWindow protocol.ByteCount,
|
|
queueWindowUpdate func(protocol.StreamID),
|
|
rttStats *utils.RTTStats,
|
|
logger utils.Logger,
|
|
) StreamFlowController {
|
|
return &streamFlowController{
|
|
streamID: streamID,
|
|
connection: cfc.(connectionFlowControllerI),
|
|
queueWindowUpdate: func() { queueWindowUpdate(streamID) },
|
|
baseFlowController: baseFlowController{
|
|
rttStats: rttStats,
|
|
receiveWindow: receiveWindow,
|
|
receiveWindowSize: receiveWindow,
|
|
maxReceiveWindowSize: maxReceiveWindow,
|
|
sendWindow: initialSendWindow,
|
|
logger: logger,
|
|
},
|
|
}
|
|
}
|
|
|
|
// UpdateHighestReceived updates the highestReceived value, if the offset is higher.
|
|
func (c *streamFlowController) UpdateHighestReceived(offset protocol.ByteCount, final bool) error {
|
|
// If the final offset for this stream is already known, check for consistency.
|
|
if c.receivedFinalOffset {
|
|
// If we receive another final offset, check that it's the same.
|
|
if final && offset != c.highestReceived {
|
|
return &qerr.TransportError{
|
|
ErrorCode: qerr.FinalSizeError,
|
|
ErrorMessage: fmt.Sprintf("received inconsistent final offset for stream %d (old: %d, new: %d bytes)", c.streamID, c.highestReceived, offset),
|
|
}
|
|
}
|
|
// Check that the offset is below the final offset.
|
|
if offset > c.highestReceived {
|
|
return &qerr.TransportError{
|
|
ErrorCode: qerr.FinalSizeError,
|
|
ErrorMessage: fmt.Sprintf("received offset %d for stream %d, but final offset was already received at %d", offset, c.streamID, c.highestReceived),
|
|
}
|
|
}
|
|
}
|
|
|
|
if final {
|
|
c.receivedFinalOffset = true
|
|
}
|
|
if offset == c.highestReceived {
|
|
return nil
|
|
}
|
|
// A higher offset was received before.
|
|
// This can happen due to reordering.
|
|
if offset <= c.highestReceived {
|
|
if final {
|
|
return &qerr.TransportError{
|
|
ErrorCode: qerr.FinalSizeError,
|
|
ErrorMessage: fmt.Sprintf("received final offset %d for stream %d, but already received offset %d before", offset, c.streamID, c.highestReceived),
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
increment := offset - c.highestReceived
|
|
c.highestReceived = offset
|
|
if c.checkFlowControlViolation() {
|
|
return &qerr.TransportError{
|
|
ErrorCode: qerr.FlowControlError,
|
|
ErrorMessage: fmt.Sprintf("received %d bytes on stream %d, allowed %d bytes", offset, c.streamID, c.receiveWindow),
|
|
}
|
|
}
|
|
return c.connection.IncrementHighestReceived(increment)
|
|
}
|
|
|
|
func (c *streamFlowController) AddBytesRead(n protocol.ByteCount) {
|
|
c.mutex.Lock()
|
|
c.baseFlowController.addBytesRead(n)
|
|
shouldQueueWindowUpdate := c.shouldQueueWindowUpdate()
|
|
c.mutex.Unlock()
|
|
if shouldQueueWindowUpdate {
|
|
c.queueWindowUpdate()
|
|
}
|
|
c.connection.AddBytesRead(n)
|
|
}
|
|
|
|
func (c *streamFlowController) Abandon() {
|
|
c.mutex.Lock()
|
|
unread := c.highestReceived - c.bytesRead
|
|
c.mutex.Unlock()
|
|
if unread > 0 {
|
|
c.connection.AddBytesRead(unread)
|
|
}
|
|
}
|
|
|
|
func (c *streamFlowController) AddBytesSent(n protocol.ByteCount) {
|
|
c.baseFlowController.AddBytesSent(n)
|
|
c.connection.AddBytesSent(n)
|
|
}
|
|
|
|
func (c *streamFlowController) SendWindowSize() protocol.ByteCount {
|
|
return utils.Min(c.baseFlowController.sendWindowSize(), c.connection.SendWindowSize())
|
|
}
|
|
|
|
func (c *streamFlowController) shouldQueueWindowUpdate() bool {
|
|
return !c.receivedFinalOffset && c.hasWindowUpdate()
|
|
}
|
|
|
|
func (c *streamFlowController) GetWindowUpdate() protocol.ByteCount {
|
|
// If we already received the final offset for this stream, the peer won't need any additional flow control credit.
|
|
if c.receivedFinalOffset {
|
|
return 0
|
|
}
|
|
|
|
// Don't use defer for unlocking the mutex here, GetWindowUpdate() is called frequently and defer shows up in the profiler
|
|
c.mutex.Lock()
|
|
oldWindowSize := c.receiveWindowSize
|
|
offset := c.baseFlowController.getWindowUpdate()
|
|
if c.receiveWindowSize > oldWindowSize { // auto-tuning enlarged the window size
|
|
c.logger.Debugf("Increasing receive flow control window for stream %d to %d kB", c.streamID, c.receiveWindowSize/(1<<10))
|
|
c.connection.EnsureMinimumWindowSize(protocol.ByteCount(float64(c.receiveWindowSize) * protocol.ConnectionFlowControlMultiplier))
|
|
}
|
|
c.mutex.Unlock()
|
|
return offset
|
|
}
|