mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 20:57:36 +03:00
762 lines
22 KiB
Go
762 lines
22 KiB
Go
package quic
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"fmt"
|
|
"net"
|
|
"time"
|
|
|
|
"github.com/lucas-clemente/quic-go/internal/qerr"
|
|
|
|
"github.com/lucas-clemente/quic-go/internal/ackhandler"
|
|
"github.com/lucas-clemente/quic-go/internal/handshake"
|
|
"github.com/lucas-clemente/quic-go/internal/protocol"
|
|
"github.com/lucas-clemente/quic-go/internal/utils"
|
|
"github.com/lucas-clemente/quic-go/internal/wire"
|
|
)
|
|
|
|
type packer interface {
|
|
PackCoalescedPacket(protocol.ByteCount) (*coalescedPacket, error)
|
|
PackPacket() (*packedPacket, error)
|
|
MaybePackProbePacket(protocol.EncryptionLevel) (*packedPacket, error)
|
|
MaybePackAckPacket(handshakeConfirmed bool) (*packedPacket, error)
|
|
PackConnectionClose(*qerr.QuicError) (*coalescedPacket, error)
|
|
|
|
HandleTransportParameters(*wire.TransportParameters)
|
|
SetToken([]byte)
|
|
}
|
|
|
|
type sealer interface {
|
|
handshake.LongHeaderSealer
|
|
}
|
|
|
|
type payload struct {
|
|
frames []ackhandler.Frame
|
|
ack *wire.AckFrame
|
|
length protocol.ByteCount
|
|
}
|
|
|
|
type packedPacket struct {
|
|
buffer *packetBuffer
|
|
*packetContents
|
|
}
|
|
|
|
type packetContents struct {
|
|
header *wire.ExtendedHeader
|
|
ack *wire.AckFrame
|
|
frames []ackhandler.Frame
|
|
|
|
length protocol.ByteCount
|
|
}
|
|
|
|
type coalescedPacket struct {
|
|
buffer *packetBuffer
|
|
|
|
packets []*packetContents
|
|
}
|
|
|
|
func (p *packetContents) EncryptionLevel() protocol.EncryptionLevel {
|
|
if !p.header.IsLongHeader {
|
|
return protocol.Encryption1RTT
|
|
}
|
|
//nolint:exhaustive // Will never be called for Retry packets (and they don't have encrypted data).
|
|
switch p.header.Type {
|
|
case protocol.PacketTypeInitial:
|
|
return protocol.EncryptionInitial
|
|
case protocol.PacketTypeHandshake:
|
|
return protocol.EncryptionHandshake
|
|
case protocol.PacketType0RTT:
|
|
return protocol.Encryption0RTT
|
|
default:
|
|
panic("can't determine encryption level")
|
|
}
|
|
}
|
|
|
|
func (p *packetContents) IsAckEliciting() bool {
|
|
return ackhandler.HasAckElicitingFrames(p.frames)
|
|
}
|
|
|
|
func (p *packetContents) ToAckHandlerPacket(now time.Time, q *retransmissionQueue) *ackhandler.Packet {
|
|
largestAcked := protocol.InvalidPacketNumber
|
|
if p.ack != nil {
|
|
largestAcked = p.ack.LargestAcked()
|
|
}
|
|
encLevel := p.EncryptionLevel()
|
|
for i := range p.frames {
|
|
if p.frames[i].OnLost != nil {
|
|
continue
|
|
}
|
|
switch encLevel {
|
|
case protocol.EncryptionInitial:
|
|
p.frames[i].OnLost = q.AddInitial
|
|
case protocol.EncryptionHandshake:
|
|
p.frames[i].OnLost = q.AddHandshake
|
|
case protocol.Encryption0RTT, protocol.Encryption1RTT:
|
|
p.frames[i].OnLost = q.AddAppData
|
|
}
|
|
}
|
|
return &ackhandler.Packet{
|
|
PacketNumber: p.header.PacketNumber,
|
|
LargestAcked: largestAcked,
|
|
Frames: p.frames,
|
|
Length: p.length,
|
|
EncryptionLevel: encLevel,
|
|
SendTime: now,
|
|
}
|
|
}
|
|
|
|
func getMaxPacketSize(addr net.Addr) protocol.ByteCount {
|
|
maxSize := protocol.ByteCount(protocol.MinInitialPacketSize)
|
|
// If this is not a UDP address, we don't know anything about the MTU.
|
|
// Use the minimum size of an Initial packet as the max packet size.
|
|
if udpAddr, ok := addr.(*net.UDPAddr); ok {
|
|
if utils.IsIPv4(udpAddr.IP) {
|
|
maxSize = protocol.MaxPacketSizeIPv4
|
|
} else {
|
|
maxSize = protocol.MaxPacketSizeIPv6
|
|
}
|
|
}
|
|
return maxSize
|
|
}
|
|
|
|
type packetNumberManager interface {
|
|
PeekPacketNumber(protocol.EncryptionLevel) (protocol.PacketNumber, protocol.PacketNumberLen)
|
|
PopPacketNumber(protocol.EncryptionLevel) protocol.PacketNumber
|
|
}
|
|
|
|
type sealingManager interface {
|
|
GetInitialSealer() (handshake.LongHeaderSealer, error)
|
|
GetHandshakeSealer() (handshake.LongHeaderSealer, error)
|
|
Get0RTTSealer() (handshake.LongHeaderSealer, error)
|
|
Get1RTTSealer() (handshake.ShortHeaderSealer, error)
|
|
}
|
|
|
|
type frameSource interface {
|
|
HasData() bool
|
|
AppendStreamFrames([]ackhandler.Frame, protocol.ByteCount) ([]ackhandler.Frame, protocol.ByteCount)
|
|
AppendControlFrames([]ackhandler.Frame, protocol.ByteCount) ([]ackhandler.Frame, protocol.ByteCount)
|
|
}
|
|
|
|
type ackFrameSource interface {
|
|
GetAckFrame(encLevel protocol.EncryptionLevel, onlyIfQueued bool) *wire.AckFrame
|
|
}
|
|
|
|
type packetPacker struct {
|
|
srcConnID protocol.ConnectionID
|
|
getDestConnID func() protocol.ConnectionID
|
|
|
|
perspective protocol.Perspective
|
|
version protocol.VersionNumber
|
|
cryptoSetup sealingManager
|
|
|
|
initialStream cryptoStream
|
|
handshakeStream cryptoStream
|
|
|
|
token []byte
|
|
|
|
pnManager packetNumberManager
|
|
framer frameSource
|
|
acks ackFrameSource
|
|
retransmissionQueue *retransmissionQueue
|
|
|
|
maxPacketSize protocol.ByteCount
|
|
numNonAckElicitingAcks int
|
|
}
|
|
|
|
var _ packer = &packetPacker{}
|
|
|
|
func newPacketPacker(
|
|
srcConnID protocol.ConnectionID,
|
|
getDestConnID func() protocol.ConnectionID,
|
|
initialStream cryptoStream,
|
|
handshakeStream cryptoStream,
|
|
packetNumberManager packetNumberManager,
|
|
retransmissionQueue *retransmissionQueue,
|
|
remoteAddr net.Addr, // only used for determining the max packet size
|
|
cryptoSetup sealingManager,
|
|
framer frameSource,
|
|
acks ackFrameSource,
|
|
perspective protocol.Perspective,
|
|
version protocol.VersionNumber,
|
|
) *packetPacker {
|
|
return &packetPacker{
|
|
cryptoSetup: cryptoSetup,
|
|
getDestConnID: getDestConnID,
|
|
srcConnID: srcConnID,
|
|
initialStream: initialStream,
|
|
handshakeStream: handshakeStream,
|
|
retransmissionQueue: retransmissionQueue,
|
|
perspective: perspective,
|
|
version: version,
|
|
framer: framer,
|
|
acks: acks,
|
|
pnManager: packetNumberManager,
|
|
maxPacketSize: getMaxPacketSize(remoteAddr),
|
|
}
|
|
}
|
|
|
|
// PackConnectionClose packs a packet that ONLY contains a ConnectionCloseFrame
|
|
func (p *packetPacker) PackConnectionClose(quicErr *qerr.QuicError) (*coalescedPacket, error) {
|
|
var reason string
|
|
// don't send details of crypto errors
|
|
if !quicErr.IsCryptoError() {
|
|
reason = quicErr.ErrorMessage
|
|
}
|
|
|
|
buffer := getPacketBuffer()
|
|
contents := make([]*packetContents, 0, 1)
|
|
for _, encLevel := range []protocol.EncryptionLevel{protocol.EncryptionInitial, protocol.EncryptionHandshake, protocol.Encryption0RTT, protocol.Encryption1RTT} {
|
|
if p.perspective == protocol.PerspectiveServer && encLevel == protocol.Encryption0RTT {
|
|
continue
|
|
}
|
|
quicErrToSend := quicErr
|
|
reasonPhrase := reason
|
|
if encLevel == protocol.EncryptionInitial || encLevel == protocol.EncryptionHandshake {
|
|
// don't send application errors in Initial or Handshake packets
|
|
if quicErr.IsApplicationError() {
|
|
quicErrToSend = qerr.NewError(qerr.ApplicationError, "")
|
|
reasonPhrase = ""
|
|
}
|
|
}
|
|
ccf := &wire.ConnectionCloseFrame{
|
|
IsApplicationError: quicErrToSend.IsApplicationError(),
|
|
ErrorCode: quicErrToSend.ErrorCode,
|
|
FrameType: quicErrToSend.FrameType,
|
|
ReasonPhrase: reasonPhrase,
|
|
}
|
|
payload := payload{
|
|
frames: []ackhandler.Frame{{Frame: ccf}},
|
|
length: ccf.Length(p.version),
|
|
}
|
|
|
|
var sealer sealer
|
|
var err error
|
|
var keyPhase protocol.KeyPhaseBit // only set for 1-RTT
|
|
switch encLevel {
|
|
case protocol.EncryptionInitial:
|
|
sealer, err = p.cryptoSetup.GetInitialSealer()
|
|
case protocol.EncryptionHandshake:
|
|
sealer, err = p.cryptoSetup.GetHandshakeSealer()
|
|
case protocol.Encryption0RTT:
|
|
sealer, err = p.cryptoSetup.Get0RTTSealer()
|
|
case protocol.Encryption1RTT:
|
|
var s handshake.ShortHeaderSealer
|
|
s, err = p.cryptoSetup.Get1RTTSealer()
|
|
if err == nil {
|
|
keyPhase = s.KeyPhase()
|
|
}
|
|
sealer = s
|
|
}
|
|
if err == handshake.ErrKeysNotYetAvailable || err == handshake.ErrKeysDropped {
|
|
continue
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var hdr *wire.ExtendedHeader
|
|
if encLevel == protocol.Encryption1RTT {
|
|
hdr = p.getShortHeader(keyPhase)
|
|
} else {
|
|
hdr = p.getLongHeader(encLevel)
|
|
}
|
|
c, err := p.appendPacket(buffer, hdr, payload, encLevel, sealer)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
contents = append(contents, c)
|
|
}
|
|
|
|
if p.perspective == protocol.PerspectiveClient && contents[0].header.Type == protocol.PacketTypeInitial {
|
|
p.padPacket(buffer)
|
|
}
|
|
|
|
return &coalescedPacket{buffer: buffer, packets: contents}, nil
|
|
}
|
|
|
|
func (p *packetPacker) MaybePackAckPacket(handshakeConfirmed bool) (*packedPacket, error) {
|
|
var encLevel protocol.EncryptionLevel
|
|
var ack *wire.AckFrame
|
|
if !handshakeConfirmed {
|
|
ack = p.acks.GetAckFrame(protocol.EncryptionInitial, true)
|
|
if ack != nil {
|
|
encLevel = protocol.EncryptionInitial
|
|
} else {
|
|
ack = p.acks.GetAckFrame(protocol.EncryptionHandshake, true)
|
|
if ack != nil {
|
|
encLevel = protocol.EncryptionHandshake
|
|
}
|
|
}
|
|
}
|
|
if ack == nil {
|
|
ack = p.acks.GetAckFrame(protocol.Encryption1RTT, true)
|
|
if ack == nil {
|
|
return nil, nil
|
|
}
|
|
encLevel = protocol.Encryption1RTT
|
|
}
|
|
if ack == nil {
|
|
return nil, nil
|
|
}
|
|
payload := payload{
|
|
ack: ack,
|
|
length: ack.Length(p.version),
|
|
}
|
|
|
|
sealer, hdr, err := p.getSealerAndHeader(encLevel)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return p.writeSinglePacket(hdr, payload, encLevel, sealer)
|
|
}
|
|
|
|
func (p *packetPacker) padPacket(buffer *packetBuffer) {
|
|
if dataLen := protocol.ByteCount(len(buffer.Data)); dataLen < p.maxPacketSize {
|
|
buffer.Data = buffer.Data[:p.maxPacketSize]
|
|
for n := dataLen; n < p.maxPacketSize; n++ {
|
|
buffer.Data[n] = 0
|
|
}
|
|
}
|
|
}
|
|
|
|
// PackCoalescedPacket packs a new packet.
|
|
// It packs an Initial / Handshake if there is data to send in these packet number spaces.
|
|
// It should only be called before the handshake is confirmed.
|
|
func (p *packetPacker) PackCoalescedPacket(maxPacketSize protocol.ByteCount) (*coalescedPacket, error) {
|
|
buffer := getPacketBuffer()
|
|
packet, err := p.packCoalescedPacket(buffer, maxPacketSize)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if packet == nil || len(packet.packets) == 0 { // nothing to send
|
|
buffer.Release()
|
|
return nil, nil
|
|
}
|
|
|
|
if p.perspective == protocol.PerspectiveClient && packet.packets[0].header.Type == protocol.PacketTypeInitial {
|
|
p.padPacket(buffer)
|
|
}
|
|
|
|
return packet, nil
|
|
}
|
|
|
|
func (p *packetPacker) packCoalescedPacket(buffer *packetBuffer, maxPacketSize protocol.ByteCount) (*coalescedPacket, error) {
|
|
maxPacketSize = utils.MinByteCount(maxPacketSize, p.maxPacketSize)
|
|
if p.perspective == protocol.PerspectiveClient {
|
|
maxPacketSize = protocol.MinInitialPacketSize
|
|
}
|
|
if maxPacketSize < protocol.MinCoalescedPacketSize {
|
|
return nil, nil
|
|
}
|
|
|
|
packet := &coalescedPacket{
|
|
buffer: buffer,
|
|
packets: make([]*packetContents, 0, 3),
|
|
}
|
|
// Try packing an Initial packet.
|
|
contents, err := p.maybeAppendCryptoPacket(buffer, maxPacketSize, protocol.EncryptionInitial)
|
|
if err != nil && err != handshake.ErrKeysDropped {
|
|
return nil, err
|
|
}
|
|
if contents != nil {
|
|
packet.packets = append(packet.packets, contents)
|
|
}
|
|
if buffer.Len() >= maxPacketSize-protocol.MinCoalescedPacketSize {
|
|
return packet, nil
|
|
}
|
|
|
|
// Add a Handshake packet.
|
|
contents, err = p.maybeAppendCryptoPacket(buffer, maxPacketSize, protocol.EncryptionHandshake)
|
|
if err != nil && err != handshake.ErrKeysDropped && err != handshake.ErrKeysNotYetAvailable {
|
|
return nil, err
|
|
}
|
|
if contents != nil {
|
|
packet.packets = append(packet.packets, contents)
|
|
}
|
|
if buffer.Len() >= maxPacketSize-protocol.MinCoalescedPacketSize {
|
|
return packet, nil
|
|
}
|
|
|
|
// Add a 0-RTT / 1-RTT packet.
|
|
contents, err = p.maybeAppendAppDataPacket(buffer, maxPacketSize)
|
|
if err == handshake.ErrKeysNotYetAvailable {
|
|
return packet, nil
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if contents != nil {
|
|
packet.packets = append(packet.packets, contents)
|
|
}
|
|
return packet, nil
|
|
}
|
|
|
|
// PackPacket packs a packet in the application data packet number space.
|
|
// It should be called after the handshake is confirmed.
|
|
func (p *packetPacker) PackPacket() (*packedPacket, error) {
|
|
buffer := getPacketBuffer()
|
|
contents, err := p.maybeAppendAppDataPacket(buffer, p.maxPacketSize)
|
|
if err != nil || contents == nil {
|
|
buffer.Release()
|
|
return nil, err
|
|
}
|
|
return &packedPacket{
|
|
buffer: buffer,
|
|
packetContents: contents,
|
|
}, nil
|
|
}
|
|
|
|
func (p *packetPacker) maybeAppendCryptoPacket(buffer *packetBuffer, maxPacketSize protocol.ByteCount, encLevel protocol.EncryptionLevel) (*packetContents, error) {
|
|
var sealer sealer
|
|
var s cryptoStream
|
|
var hasRetransmission bool
|
|
//nolint:exhaustive // Initial and Handshake are the only two encryption levels here.
|
|
switch encLevel {
|
|
case protocol.EncryptionInitial:
|
|
s = p.initialStream
|
|
hasRetransmission = p.retransmissionQueue.HasInitialData()
|
|
var err error
|
|
sealer, err = p.cryptoSetup.GetInitialSealer()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
case protocol.EncryptionHandshake:
|
|
s = p.handshakeStream
|
|
hasRetransmission = p.retransmissionQueue.HasHandshakeData()
|
|
var err error
|
|
sealer, err = p.cryptoSetup.GetHandshakeSealer()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
hasData := s.HasData()
|
|
var ack *wire.AckFrame
|
|
if encLevel != protocol.EncryptionHandshake || buffer.Len() == 0 {
|
|
ack = p.acks.GetAckFrame(encLevel, !hasRetransmission && !hasData)
|
|
}
|
|
if !hasData && !hasRetransmission && ack == nil {
|
|
// nothing to send
|
|
return nil, nil
|
|
}
|
|
|
|
remainingLen := maxPacketSize - buffer.Len() - protocol.ByteCount(sealer.Overhead())
|
|
|
|
var payload payload
|
|
if ack != nil {
|
|
payload.ack = ack
|
|
payload.length = ack.Length(p.version)
|
|
remainingLen -= payload.length
|
|
}
|
|
hdr := p.getLongHeader(encLevel)
|
|
remainingLen -= hdr.GetLength(p.version)
|
|
if hasRetransmission {
|
|
for {
|
|
var f wire.Frame
|
|
//nolint:exhaustive // 0-RTT packets can't contain any retransmission.s
|
|
switch encLevel {
|
|
case protocol.EncryptionInitial:
|
|
f = p.retransmissionQueue.GetInitialFrame(remainingLen)
|
|
case protocol.EncryptionHandshake:
|
|
f = p.retransmissionQueue.GetHandshakeFrame(remainingLen)
|
|
}
|
|
if f == nil {
|
|
break
|
|
}
|
|
payload.frames = append(payload.frames, ackhandler.Frame{Frame: f})
|
|
frameLen := f.Length(p.version)
|
|
payload.length += frameLen
|
|
remainingLen -= frameLen
|
|
}
|
|
} else if s.HasData() {
|
|
cf := s.PopCryptoFrame(remainingLen)
|
|
payload.frames = []ackhandler.Frame{{Frame: cf}}
|
|
payload.length += cf.Length(p.version)
|
|
}
|
|
return p.appendPacket(buffer, hdr, payload, encLevel, sealer)
|
|
}
|
|
|
|
func (p *packetPacker) maybeAppendAppDataPacket(buffer *packetBuffer, maxPacketSize protocol.ByteCount) (*packetContents, error) {
|
|
var sealer sealer
|
|
var header *wire.ExtendedHeader
|
|
var encLevel protocol.EncryptionLevel
|
|
oneRTTSealer, err := p.cryptoSetup.Get1RTTSealer()
|
|
if err == nil {
|
|
encLevel = protocol.Encryption1RTT
|
|
sealer = oneRTTSealer
|
|
header = p.getShortHeader(oneRTTSealer.KeyPhase())
|
|
} else {
|
|
// 1-RTT sealer not yet available
|
|
if p.perspective != protocol.PerspectiveClient {
|
|
return nil, nil
|
|
}
|
|
sealer, err = p.cryptoSetup.Get0RTTSealer()
|
|
if sealer == nil || err != nil {
|
|
return nil, nil
|
|
}
|
|
encLevel = protocol.Encryption0RTT
|
|
header = p.getLongHeader(protocol.Encryption0RTT)
|
|
}
|
|
headerLen := header.GetLength(p.version)
|
|
|
|
maxSize := maxPacketSize - buffer.Len() - protocol.ByteCount(sealer.Overhead()) - headerLen
|
|
payload := p.composeNextPacket(maxSize, encLevel == protocol.Encryption1RTT && buffer.Len() == 0)
|
|
|
|
// check if we have anything to send
|
|
if len(payload.frames) == 0 && payload.ack == nil {
|
|
return nil, nil
|
|
}
|
|
if len(payload.frames) == 0 { // the packet only contains an ACK
|
|
if p.numNonAckElicitingAcks >= protocol.MaxNonAckElicitingAcks {
|
|
ping := &wire.PingFrame{}
|
|
payload.frames = append(payload.frames, ackhandler.Frame{Frame: ping})
|
|
payload.length += ping.Length(p.version)
|
|
p.numNonAckElicitingAcks = 0
|
|
} else {
|
|
p.numNonAckElicitingAcks++
|
|
}
|
|
} else {
|
|
p.numNonAckElicitingAcks = 0
|
|
}
|
|
|
|
return p.appendPacket(buffer, header, payload, encLevel, sealer)
|
|
}
|
|
|
|
func (p *packetPacker) composeNextPacket(maxFrameSize protocol.ByteCount, ackAllowed bool) payload {
|
|
var payload payload
|
|
var ack *wire.AckFrame
|
|
hasData := p.framer.HasData()
|
|
hasRetransmission := p.retransmissionQueue.HasAppData()
|
|
if ackAllowed {
|
|
ack = p.acks.GetAckFrame(protocol.Encryption1RTT, !hasRetransmission && !hasData)
|
|
if ack != nil {
|
|
payload.ack = ack
|
|
payload.length += ack.Length(p.version)
|
|
}
|
|
}
|
|
|
|
if ack == nil && !hasData && !hasRetransmission {
|
|
return payload
|
|
}
|
|
|
|
if hasRetransmission {
|
|
for {
|
|
remainingLen := maxFrameSize - payload.length
|
|
if remainingLen < protocol.MinStreamFrameSize {
|
|
break
|
|
}
|
|
f := p.retransmissionQueue.GetAppDataFrame(remainingLen)
|
|
if f == nil {
|
|
break
|
|
}
|
|
payload.frames = append(payload.frames, ackhandler.Frame{Frame: f})
|
|
payload.length += f.Length(p.version)
|
|
}
|
|
}
|
|
|
|
if hasData {
|
|
var lengthAdded protocol.ByteCount
|
|
payload.frames, lengthAdded = p.framer.AppendControlFrames(payload.frames, maxFrameSize-payload.length)
|
|
payload.length += lengthAdded
|
|
|
|
payload.frames, lengthAdded = p.framer.AppendStreamFrames(payload.frames, maxFrameSize-payload.length)
|
|
payload.length += lengthAdded
|
|
}
|
|
return payload
|
|
}
|
|
|
|
func (p *packetPacker) MaybePackProbePacket(encLevel protocol.EncryptionLevel) (*packedPacket, error) {
|
|
var contents *packetContents
|
|
var err error
|
|
buffer := getPacketBuffer()
|
|
//nolint:exhaustive Probe packets are never sent for 0-RTT.
|
|
switch encLevel {
|
|
case protocol.EncryptionInitial:
|
|
contents, err = p.maybeAppendCryptoPacket(buffer, p.maxPacketSize, protocol.EncryptionInitial)
|
|
case protocol.EncryptionHandshake:
|
|
contents, err = p.maybeAppendCryptoPacket(buffer, p.maxPacketSize, protocol.EncryptionHandshake)
|
|
case protocol.Encryption1RTT:
|
|
contents, err = p.maybeAppendAppDataPacket(buffer, p.maxPacketSize)
|
|
default:
|
|
panic("unknown encryption level")
|
|
}
|
|
if err != nil || contents == nil {
|
|
return nil, err
|
|
}
|
|
if p.perspective == protocol.PerspectiveClient && encLevel == protocol.EncryptionInitial {
|
|
p.padPacket(buffer)
|
|
}
|
|
return &packedPacket{
|
|
buffer: buffer,
|
|
packetContents: contents,
|
|
}, nil
|
|
}
|
|
|
|
func (p *packetPacker) getSealerAndHeader(encLevel protocol.EncryptionLevel) (sealer, *wire.ExtendedHeader, error) {
|
|
switch encLevel {
|
|
case protocol.EncryptionInitial:
|
|
sealer, err := p.cryptoSetup.GetInitialSealer()
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
hdr := p.getLongHeader(protocol.EncryptionInitial)
|
|
return sealer, hdr, nil
|
|
case protocol.Encryption0RTT:
|
|
sealer, err := p.cryptoSetup.Get0RTTSealer()
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
hdr := p.getLongHeader(protocol.Encryption0RTT)
|
|
return sealer, hdr, nil
|
|
case protocol.EncryptionHandshake:
|
|
sealer, err := p.cryptoSetup.GetHandshakeSealer()
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
hdr := p.getLongHeader(protocol.EncryptionHandshake)
|
|
return sealer, hdr, nil
|
|
case protocol.Encryption1RTT:
|
|
sealer, err := p.cryptoSetup.Get1RTTSealer()
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
hdr := p.getShortHeader(sealer.KeyPhase())
|
|
return sealer, hdr, nil
|
|
default:
|
|
return nil, nil, fmt.Errorf("unexpected encryption level: %s", encLevel)
|
|
}
|
|
}
|
|
|
|
func (p *packetPacker) getShortHeader(kp protocol.KeyPhaseBit) *wire.ExtendedHeader {
|
|
pn, pnLen := p.pnManager.PeekPacketNumber(protocol.Encryption1RTT)
|
|
hdr := &wire.ExtendedHeader{}
|
|
hdr.PacketNumber = pn
|
|
hdr.PacketNumberLen = pnLen
|
|
hdr.DestConnectionID = p.getDestConnID()
|
|
hdr.KeyPhase = kp
|
|
return hdr
|
|
}
|
|
|
|
func (p *packetPacker) getLongHeader(encLevel protocol.EncryptionLevel) *wire.ExtendedHeader {
|
|
pn, pnLen := p.pnManager.PeekPacketNumber(encLevel)
|
|
hdr := &wire.ExtendedHeader{}
|
|
hdr.IsLongHeader = true
|
|
hdr.Version = p.version
|
|
hdr.SrcConnectionID = p.srcConnID
|
|
hdr.DestConnectionID = p.getDestConnID()
|
|
|
|
// Set the length to the maximum packet size.
|
|
// Since it is encoded as a varint, this guarantees us that the header will end up at most as big as GetLength() returns.
|
|
hdr.Length = p.maxPacketSize
|
|
|
|
hdr.PacketNumber = pn
|
|
hdr.PacketNumberLen = pnLen
|
|
|
|
//nolint:exhaustive // 1-RTT packets are not long header packets.
|
|
switch encLevel {
|
|
case protocol.EncryptionInitial:
|
|
hdr.Type = protocol.PacketTypeInitial
|
|
hdr.Token = p.token
|
|
case protocol.EncryptionHandshake:
|
|
hdr.Type = protocol.PacketTypeHandshake
|
|
case protocol.Encryption0RTT:
|
|
hdr.Type = protocol.PacketType0RTT
|
|
}
|
|
|
|
return hdr
|
|
}
|
|
|
|
// writeSinglePacket packs a single packet.
|
|
func (p *packetPacker) writeSinglePacket(
|
|
header *wire.ExtendedHeader,
|
|
payload payload,
|
|
encLevel protocol.EncryptionLevel,
|
|
sealer sealer,
|
|
) (*packedPacket, error) {
|
|
buffer := getPacketBuffer()
|
|
contents, err := p.appendPacket(buffer, header, payload, encLevel, sealer)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &packedPacket{
|
|
buffer: buffer,
|
|
packetContents: contents,
|
|
}, nil
|
|
}
|
|
|
|
func (p *packetPacker) appendPacket(
|
|
buffer *packetBuffer,
|
|
header *wire.ExtendedHeader,
|
|
payload payload,
|
|
encLevel protocol.EncryptionLevel,
|
|
sealer sealer,
|
|
) (*packetContents, error) {
|
|
var paddingLen protocol.ByteCount
|
|
pnLen := protocol.ByteCount(header.PacketNumberLen)
|
|
if payload.length < 4-pnLen {
|
|
paddingLen = 4 - pnLen - payload.length
|
|
}
|
|
if header.IsLongHeader {
|
|
header.Length = pnLen + protocol.ByteCount(sealer.Overhead()) + payload.length + paddingLen
|
|
}
|
|
|
|
hdrOffset := buffer.Len()
|
|
buf := bytes.NewBuffer(buffer.Data)
|
|
if err := header.Write(buf, p.version); err != nil {
|
|
return nil, err
|
|
}
|
|
payloadOffset := buf.Len()
|
|
|
|
if payload.ack != nil {
|
|
if err := payload.ack.Write(buf, p.version); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
if paddingLen > 0 {
|
|
buf.Write(bytes.Repeat([]byte{0}, int(paddingLen)))
|
|
}
|
|
for _, frame := range payload.frames {
|
|
if err := frame.Write(buf, p.version); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
if payloadSize := protocol.ByteCount(buf.Len()-payloadOffset) - paddingLen; payloadSize != payload.length {
|
|
return nil, fmt.Errorf("PacketPacker BUG: payload size inconsistent (expected %d, got %d bytes)", payload.length, payloadSize)
|
|
}
|
|
if size := protocol.ByteCount(buf.Len() + sealer.Overhead()); size > p.maxPacketSize {
|
|
return nil, fmt.Errorf("PacketPacker BUG: packet too large (%d bytes, allowed %d bytes)", size, p.maxPacketSize)
|
|
}
|
|
|
|
raw := buffer.Data
|
|
// encrypt the packet
|
|
raw = raw[:buf.Len()]
|
|
_ = sealer.Seal(raw[payloadOffset:payloadOffset], raw[payloadOffset:], header.PacketNumber, raw[hdrOffset:payloadOffset])
|
|
raw = raw[0 : buf.Len()+sealer.Overhead()]
|
|
// apply header protection
|
|
pnOffset := payloadOffset - int(header.PacketNumberLen)
|
|
sealer.EncryptHeader(raw[pnOffset+4:pnOffset+4+16], &raw[hdrOffset], raw[pnOffset:payloadOffset])
|
|
buffer.Data = raw
|
|
|
|
num := p.pnManager.PopPacketNumber(encLevel)
|
|
if num != header.PacketNumber {
|
|
return nil, errors.New("packetPacker BUG: Peeked and Popped packet numbers do not match")
|
|
}
|
|
return &packetContents{
|
|
header: header,
|
|
ack: payload.ack,
|
|
frames: payload.frames,
|
|
length: buffer.Len() - hdrOffset,
|
|
}, nil
|
|
}
|
|
|
|
func (p *packetPacker) SetToken(token []byte) {
|
|
p.token = token
|
|
}
|
|
|
|
func (p *packetPacker) HandleTransportParameters(params *wire.TransportParameters) {
|
|
if params.MaxUDPPayloadSize != 0 {
|
|
p.maxPacketSize = utils.MinByteCount(p.maxPacketSize, params.MaxUDPPayloadSize)
|
|
}
|
|
}
|