92 lines
2.6 KiB
Go
92 lines
2.6 KiB
Go
|
package frames
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"errors"
|
||
|
|
||
|
"github.com/lucas-clemente/quic-go/internal/utils"
|
||
|
"github.com/lucas-clemente/quic-go/protocol"
|
||
|
"github.com/lucas-clemente/quic-go/qerr"
|
||
|
)
|
||
|
|
||
|
// A StopWaitingFrame in QUIC
|
||
|
type StopWaitingFrame struct {
|
||
|
LeastUnacked protocol.PacketNumber
|
||
|
PacketNumberLen protocol.PacketNumberLen
|
||
|
PacketNumber protocol.PacketNumber
|
||
|
}
|
||
|
|
||
|
var (
|
||
|
errLeastUnackedHigherThanPacketNumber = errors.New("StopWaitingFrame: LeastUnacked can't be greater than the packet number")
|
||
|
errPacketNumberNotSet = errors.New("StopWaitingFrame: PacketNumber not set")
|
||
|
errPacketNumberLenNotSet = errors.New("StopWaitingFrame: PacketNumberLen not set")
|
||
|
)
|
||
|
|
||
|
func (f *StopWaitingFrame) Write(b *bytes.Buffer, version protocol.VersionNumber) error {
|
||
|
// packetNumber is the packet number of the packet that this StopWaitingFrame will be sent with
|
||
|
typeByte := uint8(0x06)
|
||
|
b.WriteByte(typeByte)
|
||
|
|
||
|
// make sure the PacketNumber was set
|
||
|
if f.PacketNumber == protocol.PacketNumber(0) {
|
||
|
return errPacketNumberNotSet
|
||
|
}
|
||
|
|
||
|
if f.LeastUnacked > f.PacketNumber {
|
||
|
return errLeastUnackedHigherThanPacketNumber
|
||
|
}
|
||
|
|
||
|
leastUnackedDelta := uint64(f.PacketNumber - f.LeastUnacked)
|
||
|
|
||
|
switch f.PacketNumberLen {
|
||
|
case protocol.PacketNumberLen1:
|
||
|
b.WriteByte(uint8(leastUnackedDelta))
|
||
|
case protocol.PacketNumberLen2:
|
||
|
utils.WriteUint16(b, uint16(leastUnackedDelta))
|
||
|
case protocol.PacketNumberLen4:
|
||
|
utils.WriteUint32(b, uint32(leastUnackedDelta))
|
||
|
case protocol.PacketNumberLen6:
|
||
|
utils.WriteUint48(b, leastUnackedDelta)
|
||
|
default:
|
||
|
return errPacketNumberLenNotSet
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// MinLength of a written frame
|
||
|
func (f *StopWaitingFrame) MinLength(version protocol.VersionNumber) (protocol.ByteCount, error) {
|
||
|
minLength := protocol.ByteCount(1) // typeByte
|
||
|
|
||
|
if f.PacketNumberLen == protocol.PacketNumberLenInvalid {
|
||
|
return 0, errPacketNumberLenNotSet
|
||
|
}
|
||
|
minLength += protocol.ByteCount(f.PacketNumberLen)
|
||
|
|
||
|
return minLength, nil
|
||
|
}
|
||
|
|
||
|
// ParseStopWaitingFrame parses a StopWaiting frame
|
||
|
func ParseStopWaitingFrame(r *bytes.Reader, packetNumber protocol.PacketNumber, packetNumberLen protocol.PacketNumberLen, version protocol.VersionNumber) (*StopWaitingFrame, error) {
|
||
|
frame := &StopWaitingFrame{}
|
||
|
|
||
|
// read the TypeByte
|
||
|
_, err := r.ReadByte()
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
leastUnackedDelta, err := utils.ReadUintN(r, uint8(packetNumberLen))
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
if leastUnackedDelta > uint64(packetNumber) {
|
||
|
return nil, qerr.Error(qerr.InvalidStopWaitingData, "invalid LeastUnackedDelta")
|
||
|
}
|
||
|
|
||
|
frame.LeastUnacked = protocol.PacketNumber(uint64(packetNumber) - leastUnackedDelta)
|
||
|
|
||
|
return frame, nil
|
||
|
}
|