mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-03 20:27:35 +03:00
The queue holds all streams that have data to send. When stream.Write or stream.Close are called, a stream is added to this queue. It is removed from the queue when all available (at that moment) data was sent. This way, we don't need the round robin scheduling (which, for every packet sent, asked every single open stream if it had data) any more.
125 lines
3.6 KiB
Go
125 lines
3.6 KiB
Go
package quic
|
|
|
|
import (
|
|
"net"
|
|
"time"
|
|
|
|
"github.com/lucas-clemente/quic-go/internal/flowcontrol"
|
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
|
"github.com/lucas-clemente/quic-go/internal/wire"
|
|
)
|
|
|
|
const (
|
|
errorCodeStopping protocol.ApplicationErrorCode = 0
|
|
errorCodeStoppingGQUIC protocol.ApplicationErrorCode = 7
|
|
)
|
|
|
|
// The streamSender is notified by the stream about various events.
|
|
type streamSender interface {
|
|
queueControlFrame(wire.Frame)
|
|
onHasWindowUpdate(protocol.StreamID)
|
|
onHasStreamData(protocol.StreamID)
|
|
}
|
|
|
|
type streamI interface {
|
|
Stream
|
|
|
|
handleStreamFrame(*wire.StreamFrame) error
|
|
handleRstStreamFrame(*wire.RstStreamFrame) error
|
|
handleStopSendingFrame(*wire.StopSendingFrame)
|
|
popStreamFrame(maxBytes protocol.ByteCount) (*wire.StreamFrame, bool)
|
|
finished() bool
|
|
closeForShutdown(error)
|
|
// methods needed for flow control
|
|
getWindowUpdate() protocol.ByteCount
|
|
handleMaxStreamDataFrame(*wire.MaxStreamDataFrame)
|
|
}
|
|
|
|
// A Stream assembles the data from StreamFrames and provides a super-convenient Read-Interface
|
|
//
|
|
// Read() and Write() may be called concurrently, but multiple calls to Read() or Write() individually must be synchronized manually.
|
|
type stream struct {
|
|
receiveStream
|
|
sendStream
|
|
|
|
version protocol.VersionNumber
|
|
}
|
|
|
|
var _ Stream = &stream{}
|
|
var _ streamI = &stream{}
|
|
|
|
type deadlineError struct{}
|
|
|
|
func (deadlineError) Error() string { return "deadline exceeded" }
|
|
func (deadlineError) Temporary() bool { return true }
|
|
func (deadlineError) Timeout() bool { return true }
|
|
|
|
var errDeadline net.Error = &deadlineError{}
|
|
|
|
type streamCanceledError struct {
|
|
error
|
|
errorCode protocol.ApplicationErrorCode
|
|
}
|
|
|
|
func (streamCanceledError) Canceled() bool { return true }
|
|
func (e streamCanceledError) ErrorCode() protocol.ApplicationErrorCode { return e.errorCode }
|
|
|
|
var _ StreamError = &streamCanceledError{}
|
|
|
|
// newStream creates a new Stream
|
|
func newStream(streamID protocol.StreamID,
|
|
sender streamSender,
|
|
flowController flowcontrol.StreamFlowController,
|
|
version protocol.VersionNumber,
|
|
) *stream {
|
|
return &stream{
|
|
sendStream: *newSendStream(streamID, sender, flowController, version),
|
|
receiveStream: *newReceiveStream(streamID, sender, flowController),
|
|
}
|
|
}
|
|
|
|
// need to define StreamID() here, since both receiveStream and readStream have a StreamID()
|
|
func (s *stream) StreamID() protocol.StreamID {
|
|
// the result is same for receiveStream and sendStream
|
|
return s.sendStream.StreamID()
|
|
}
|
|
|
|
func (s *stream) Close() error {
|
|
if err := s.sendStream.Close(); err != nil {
|
|
return err
|
|
}
|
|
// in gQUIC, we need to send a RST_STREAM with the final offset if CancelRead() was called
|
|
s.receiveStream.onClose(s.sendStream.getWriteOffset())
|
|
return nil
|
|
}
|
|
|
|
func (s *stream) SetDeadline(t time.Time) error {
|
|
_ = s.SetReadDeadline(t) // SetReadDeadline never errors
|
|
_ = s.SetWriteDeadline(t) // SetWriteDeadline never errors
|
|
return nil
|
|
}
|
|
|
|
// CloseForShutdown closes a stream abruptly.
|
|
// It makes Read and Write unblock (and return the error) immediately.
|
|
// The peer will NOT be informed about this: the stream is closed without sending a FIN or RST.
|
|
func (s *stream) closeForShutdown(err error) {
|
|
s.sendStream.closeForShutdown(err)
|
|
s.receiveStream.closeForShutdown(err)
|
|
}
|
|
|
|
func (s *stream) handleRstStreamFrame(frame *wire.RstStreamFrame) error {
|
|
if err := s.receiveStream.handleRstStreamFrame(frame); err != nil {
|
|
return err
|
|
}
|
|
if !s.version.UsesIETFFrameFormat() {
|
|
s.handleStopSendingFrame(&wire.StopSendingFrame{
|
|
StreamID: s.StreamID(),
|
|
ErrorCode: frame.ErrorCode,
|
|
})
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *stream) finished() bool {
|
|
return s.sendStream.finished() && s.receiveStream.finished()
|
|
}
|