2017-05-27 14:30:11 -05:00
package quic
import (
"bytes"
"errors"
"fmt"
2018-02-17 00:29:53 -05:00
"sync"
2017-05-27 14:30:11 -05:00
2018-02-17 00:29:53 -05:00
"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/wire"
2017-05-27 14:30:11 -05:00
)
type packedPacket struct {
2018-02-17 00:29:53 -05:00
header * wire . Header
2017-05-27 14:30:11 -05:00
raw [ ] byte
2018-02-17 00:29:53 -05:00
frames [ ] wire . Frame
2017-05-27 14:30:11 -05:00
encryptionLevel protocol . EncryptionLevel
}
2018-02-17 00:29:53 -05:00
type streamFrameSource interface {
HasCryptoStreamData ( ) bool
PopCryptoStreamFrame ( protocol . ByteCount ) * wire . StreamFrame
PopStreamFrames ( protocol . ByteCount ) [ ] * wire . StreamFrame
}
2017-05-27 14:30:11 -05:00
type packetPacker struct {
connectionID protocol . ConnectionID
perspective protocol . Perspective
version protocol . VersionNumber
cryptoSetup handshake . CryptoSetup
packetNumberGenerator * packetNumberGenerator
2018-02-17 00:29:53 -05:00
streams streamFrameSource
controlFrameMutex sync . Mutex
controlFrames [ ] wire . Frame
2017-05-27 14:30:11 -05:00
2018-02-17 00:29:53 -05:00
stopWaiting * wire . StopWaitingFrame
ackFrame * wire . AckFrame
leastUnacked protocol . PacketNumber
omitConnectionID bool
hasSentPacket bool // has the packetPacker already sent a packet
numNonRetransmittableAcks int
2017-05-27 14:30:11 -05:00
}
2017-07-27 17:11:56 -05:00
func newPacketPacker ( connectionID protocol . ConnectionID ,
2018-02-17 00:29:53 -05:00
initialPacketNumber protocol . PacketNumber ,
2017-07-27 17:11:56 -05:00
cryptoSetup handshake . CryptoSetup ,
2018-02-17 00:29:53 -05:00
streamFramer streamFrameSource ,
2017-07-27 17:11:56 -05:00
perspective protocol . Perspective ,
version protocol . VersionNumber ,
) * packetPacker {
2017-05-27 14:30:11 -05:00
return & packetPacker {
cryptoSetup : cryptoSetup ,
connectionID : connectionID ,
perspective : perspective ,
version : version ,
2018-02-17 00:29:53 -05:00
streams : streamFramer ,
packetNumberGenerator : newPacketNumberGenerator ( initialPacketNumber , protocol . SkipPacketAveragePeriodLength ) ,
2017-05-27 14:30:11 -05:00
}
}
// PackConnectionClose packs a packet that ONLY contains a ConnectionCloseFrame
2018-02-17 00:29:53 -05:00
func ( p * packetPacker ) PackConnectionClose ( ccf * wire . ConnectionCloseFrame ) ( * packedPacket , error ) {
frames := [ ] wire . Frame { ccf }
2017-07-27 17:11:56 -05:00
encLevel , sealer := p . cryptoSetup . GetSealer ( )
2018-02-17 00:29:53 -05:00
header := p . getHeader ( encLevel )
raw , err := p . writeAndSealPacket ( header , frames , sealer )
2017-07-27 17:11:56 -05:00
return & packedPacket {
2018-02-17 00:29:53 -05:00
header : header ,
2017-07-27 17:11:56 -05:00
raw : raw ,
frames : frames ,
encryptionLevel : encLevel ,
} , err
}
func ( p * packetPacker ) PackAckPacket ( ) ( * packedPacket , error ) {
if p . ackFrame == nil {
return nil , errors . New ( "packet packer BUG: no ack frame queued" )
}
encLevel , sealer := p . cryptoSetup . GetSealer ( )
2018-02-17 00:29:53 -05:00
header := p . getHeader ( encLevel )
frames := [ ] wire . Frame { p . ackFrame }
if p . stopWaiting != nil { // a STOP_WAITING will only be queued when using gQUIC
p . stopWaiting . PacketNumber = header . PacketNumber
p . stopWaiting . PacketNumberLen = header . PacketNumberLen
2017-07-27 17:11:56 -05:00
frames = append ( frames , p . stopWaiting )
p . stopWaiting = nil
}
p . ackFrame = nil
2018-02-17 00:29:53 -05:00
raw , err := p . writeAndSealPacket ( header , frames , sealer )
2017-07-27 17:11:56 -05:00
return & packedPacket {
2018-02-17 00:29:53 -05:00
header : header ,
2017-07-27 17:11:56 -05:00
raw : raw ,
frames : frames ,
encryptionLevel : encLevel ,
} , err
2017-05-27 14:30:11 -05:00
}
2017-07-27 17:11:56 -05:00
// PackHandshakeRetransmission retransmits a handshake packet, that was sent with less than forward-secure encryption
func ( p * packetPacker ) PackHandshakeRetransmission ( packet * ackhandler . Packet ) ( * packedPacket , error ) {
2017-05-27 14:30:11 -05:00
if packet . EncryptionLevel == protocol . EncryptionForwardSecure {
return nil , errors . New ( "PacketPacker BUG: forward-secure encrypted handshake packets don't need special treatment" )
}
2017-07-27 17:11:56 -05:00
sealer , err := p . cryptoSetup . GetSealerWithEncryptionLevel ( packet . EncryptionLevel )
if err != nil {
return nil , err
}
2018-02-17 00:29:53 -05:00
header := p . getHeader ( packet . EncryptionLevel )
var frames [ ] wire . Frame
if ! p . version . UsesIETFFrameFormat ( ) { // for gQUIC: pack a STOP_WAITING first
if p . stopWaiting == nil {
return nil , errors . New ( "PacketPacker BUG: Handshake retransmissions must contain a STOP_WAITING frame" )
}
swf := p . stopWaiting
swf . PacketNumber = header . PacketNumber
swf . PacketNumberLen = header . PacketNumberLen
p . stopWaiting = nil
frames = append ( [ ] wire . Frame { swf } , packet . Frames ... )
} else {
frames = packet . Frames
2017-05-27 14:30:11 -05:00
}
2018-02-17 00:29:53 -05:00
raw , err := p . writeAndSealPacket ( header , frames , sealer )
2017-07-27 17:11:56 -05:00
return & packedPacket {
2018-02-17 00:29:53 -05:00
header : header ,
2017-07-27 17:11:56 -05:00
raw : raw ,
frames : frames ,
encryptionLevel : packet . EncryptionLevel ,
} , err
2017-05-27 14:30:11 -05:00
}
// PackPacket packs a new packet
// the other controlFrames are sent in the next packet, but might be queued and sent in the next packet if the packet would overflow MaxPacketSize otherwise
2017-07-27 17:11:56 -05:00
func ( p * packetPacker ) PackPacket ( ) ( * packedPacket , error ) {
2018-02-17 00:29:53 -05:00
hasCryptoStreamFrame := p . streams . HasCryptoStreamData ( )
// if this is the first packet to be send, make sure it contains stream data
if ! p . hasSentPacket && ! hasCryptoStreamFrame {
return nil , nil
}
if hasCryptoStreamFrame {
2017-07-27 17:11:56 -05:00
return p . packCryptoPacket ( )
2017-05-27 14:30:11 -05:00
}
2017-07-27 17:11:56 -05:00
encLevel , sealer := p . cryptoSetup . GetSealer ( )
2017-05-27 14:30:11 -05:00
2018-02-17 00:29:53 -05:00
header := p . getHeader ( encLevel )
headerLength , err := header . GetLength ( p . perspective , p . version )
2017-05-27 14:30:11 -05:00
if err != nil {
return nil , err
}
2017-07-27 17:11:56 -05:00
if p . stopWaiting != nil {
2018-02-17 00:29:53 -05:00
p . stopWaiting . PacketNumber = header . PacketNumber
p . stopWaiting . PacketNumberLen = header . PacketNumberLen
2017-05-27 14:30:11 -05:00
}
2018-02-17 00:29:53 -05:00
maxSize := protocol . MaxPacketSize - protocol . ByteCount ( sealer . Overhead ( ) ) - headerLength
2017-07-27 17:11:56 -05:00
payloadFrames , err := p . composeNextPacket ( maxSize , p . canSendData ( encLevel ) )
if err != nil {
return nil , err
2017-05-27 14:30:11 -05:00
}
// Check if we have enough frames to send
if len ( payloadFrames ) == 0 {
return nil , nil
}
// Don't send out packets that only contain a StopWaitingFrame
2017-07-27 17:11:56 -05:00
if len ( payloadFrames ) == 1 && p . stopWaiting != nil {
2017-05-27 14:30:11 -05:00
return nil , nil
}
2018-02-17 00:29:53 -05:00
if p . ackFrame != nil {
// check if this packet only contains an ACK (and maybe a STOP_WAITING)
if len ( payloadFrames ) == 1 || ( p . stopWaiting != nil && len ( payloadFrames ) == 2 ) {
if p . numNonRetransmittableAcks >= protocol . MaxNonRetransmittableAcks {
payloadFrames = append ( payloadFrames , & wire . PingFrame { } )
p . numNonRetransmittableAcks = 0
} else {
p . numNonRetransmittableAcks ++
}
} else {
p . numNonRetransmittableAcks = 0
}
}
2017-07-27 17:11:56 -05:00
p . stopWaiting = nil
p . ackFrame = nil
2017-05-27 14:30:11 -05:00
2018-02-17 00:29:53 -05:00
raw , err := p . writeAndSealPacket ( header , payloadFrames , sealer )
2017-07-27 17:11:56 -05:00
if err != nil {
2017-05-27 14:30:11 -05:00
return nil , err
}
2017-07-27 17:11:56 -05:00
return & packedPacket {
2018-02-17 00:29:53 -05:00
header : header ,
2017-07-27 17:11:56 -05:00
raw : raw ,
frames : payloadFrames ,
encryptionLevel : encLevel ,
} , nil
}
2017-05-27 14:30:11 -05:00
2017-07-27 17:11:56 -05:00
func ( p * packetPacker ) packCryptoPacket ( ) ( * packedPacket , error ) {
encLevel , sealer := p . cryptoSetup . GetSealerForCryptoStream ( )
2018-02-17 00:29:53 -05:00
header := p . getHeader ( encLevel )
headerLength , err := header . GetLength ( p . perspective , p . version )
2017-07-27 17:11:56 -05:00
if err != nil {
return nil , err
2017-05-27 14:30:11 -05:00
}
2018-02-17 00:29:53 -05:00
maxLen := protocol . MaxPacketSize - protocol . ByteCount ( sealer . Overhead ( ) ) - protocol . NonForwardSecurePacketSizeReduction - headerLength
sf := p . streams . PopCryptoStreamFrame ( maxLen )
sf . DataLenPresent = false
frames := [ ] wire . Frame { sf }
raw , err := p . writeAndSealPacket ( header , frames , sealer )
2017-07-27 17:11:56 -05:00
if err != nil {
return nil , err
2017-05-27 14:30:11 -05:00
}
return & packedPacket {
2018-02-17 00:29:53 -05:00
header : header ,
2017-05-27 14:30:11 -05:00
raw : raw ,
2017-07-27 17:11:56 -05:00
frames : frames ,
2017-05-27 14:30:11 -05:00
encryptionLevel : encLevel ,
} , nil
}
2017-07-27 17:11:56 -05:00
func ( p * packetPacker ) composeNextPacket (
maxFrameSize protocol . ByteCount ,
canSendStreamFrames bool ,
2018-02-17 00:29:53 -05:00
) ( [ ] wire . Frame , error ) {
2017-05-27 14:30:11 -05:00
var payloadLength protocol . ByteCount
2018-02-17 00:29:53 -05:00
var payloadFrames [ ] wire . Frame
2017-05-27 14:30:11 -05:00
2017-07-27 17:11:56 -05:00
// STOP_WAITING and ACK will always fit
2018-02-17 00:29:53 -05:00
if p . ackFrame != nil { // ACKs need to go first, so that the sentPacketHandler will recognize them
2017-07-27 17:11:56 -05:00
payloadFrames = append ( payloadFrames , p . ackFrame )
2018-02-17 00:29:53 -05:00
l := p . ackFrame . Length ( p . version )
2017-07-27 17:11:56 -05:00
payloadLength += l
2017-05-27 14:30:11 -05:00
}
2018-02-17 00:29:53 -05:00
if p . stopWaiting != nil { // a STOP_WAITING will only be queued when using gQUIC
payloadFrames = append ( payloadFrames , p . stopWaiting )
payloadLength += p . stopWaiting . Length ( p . version )
}
2017-05-27 14:30:11 -05:00
2018-02-17 00:29:53 -05:00
p . controlFrameMutex . Lock ( )
2017-05-27 14:30:11 -05:00
for len ( p . controlFrames ) > 0 {
frame := p . controlFrames [ len ( p . controlFrames ) - 1 ]
2018-02-17 00:29:53 -05:00
length := frame . Length ( p . version )
if payloadLength + length > maxFrameSize {
2017-05-27 14:30:11 -05:00
break
}
payloadFrames = append ( payloadFrames , frame )
2018-02-17 00:29:53 -05:00
payloadLength += length
2017-05-27 14:30:11 -05:00
p . controlFrames = p . controlFrames [ : len ( p . controlFrames ) - 1 ]
}
2018-02-17 00:29:53 -05:00
p . controlFrameMutex . Unlock ( )
2017-05-27 14:30:11 -05:00
if payloadLength > maxFrameSize {
return nil , fmt . Errorf ( "Packet Packer BUG: packet payload (%d) too large (%d)" , payloadLength , maxFrameSize )
}
2017-07-27 17:11:56 -05:00
if ! canSendStreamFrames {
return payloadFrames , nil
}
2018-02-17 00:29:53 -05:00
// temporarily increase the maxFrameSize by the (minimum) length of the DataLen field
2017-05-27 14:30:11 -05:00
// this leads to a properly sized packet in all cases, since we do all the packet length calculations with StreamFrames that have the DataLen set
2018-02-17 00:29:53 -05:00
// however, for the last StreamFrame in the packet, we can omit the DataLen, thus yielding a packet of exactly the correct size
// for gQUIC STREAM frames, DataLen is always 2 bytes
// for IETF draft style STREAM frames, the length is encoded to either 1 or 2 bytes
if p . version . UsesIETFFrameFormat ( ) {
maxFrameSize ++
} else {
maxFrameSize += 2
}
2017-05-27 14:30:11 -05:00
2018-02-17 00:29:53 -05:00
fs := p . streams . PopStreamFrames ( maxFrameSize - payloadLength )
2017-05-27 14:30:11 -05:00
if len ( fs ) != 0 {
fs [ len ( fs ) - 1 ] . DataLenPresent = false
}
for _ , f := range fs {
payloadFrames = append ( payloadFrames , f )
}
return payloadFrames , nil
}
2018-02-17 00:29:53 -05:00
func ( p * packetPacker ) QueueControlFrame ( frame wire . Frame ) {
2017-07-27 17:11:56 -05:00
switch f := frame . ( type ) {
2018-02-17 00:29:53 -05:00
case * wire . StopWaitingFrame :
2017-07-27 17:11:56 -05:00
p . stopWaiting = f
2018-02-17 00:29:53 -05:00
case * wire . AckFrame :
2017-07-27 17:11:56 -05:00
p . ackFrame = f
default :
2018-02-17 00:29:53 -05:00
p . controlFrameMutex . Lock ( )
2017-07-27 17:11:56 -05:00
p . controlFrames = append ( p . controlFrames , f )
2018-02-17 00:29:53 -05:00
p . controlFrameMutex . Unlock ( )
2017-07-27 17:11:56 -05:00
}
}
2018-02-17 00:29:53 -05:00
func ( p * packetPacker ) getHeader ( encLevel protocol . EncryptionLevel ) * wire . Header {
2017-07-27 17:11:56 -05:00
pnum := p . packetNumberGenerator . Peek ( )
2018-02-17 00:29:53 -05:00
packetNumberLen := protocol . GetPacketNumberLengthForHeader ( pnum , p . leastUnacked )
2017-07-27 17:11:56 -05:00
2018-02-17 00:29:53 -05:00
header := & wire . Header {
ConnectionID : p . connectionID ,
PacketNumber : pnum ,
PacketNumberLen : packetNumberLen ,
2017-07-27 17:11:56 -05:00
}
2018-02-17 00:29:53 -05:00
if p . version . UsesTLS ( ) && encLevel != protocol . EncryptionForwardSecure {
header . PacketNumberLen = protocol . PacketNumberLen4
header . IsLongHeader = true
if ! p . hasSentPacket && p . perspective == protocol . PerspectiveClient {
header . Type = protocol . PacketTypeInitial
} else {
header . Type = protocol . PacketTypeHandshake
}
2017-07-27 17:11:56 -05:00
}
2018-02-17 00:29:53 -05:00
if p . omitConnectionID && encLevel == protocol . EncryptionForwardSecure {
header . OmitConnectionID = true
}
if ! p . version . UsesTLS ( ) {
if p . perspective == protocol . PerspectiveServer && encLevel == protocol . EncryptionSecure {
header . DiversificationNonce = p . cryptoSetup . DiversificationNonce ( )
}
if p . perspective == protocol . PerspectiveClient && encLevel != protocol . EncryptionForwardSecure {
header . VersionFlag = true
header . Version = p . version
}
} else {
if encLevel != protocol . EncryptionForwardSecure {
header . Version = p . version
}
}
return header
2017-07-27 17:11:56 -05:00
}
func ( p * packetPacker ) writeAndSealPacket (
2018-02-17 00:29:53 -05:00
header * wire . Header ,
payloadFrames [ ] wire . Frame ,
2017-07-27 17:11:56 -05:00
sealer handshake . Sealer ,
) ( [ ] byte , error ) {
raw := getPacketBuffer ( )
buffer := bytes . NewBuffer ( raw )
2018-02-17 00:29:53 -05:00
if err := header . Write ( buffer , p . perspective , p . version ) ; err != nil {
2017-07-27 17:11:56 -05:00
return nil , err
}
payloadStartIndex := buffer . Len ( )
2018-02-17 00:29:53 -05:00
// the Initial packet needs to be padded, so the last STREAM frame must have the data length present
if header . Type == protocol . PacketTypeInitial {
lastFrame := payloadFrames [ len ( payloadFrames ) - 1 ]
if sf , ok := lastFrame . ( * wire . StreamFrame ) ; ok {
sf . DataLenPresent = true
}
}
2017-07-27 17:11:56 -05:00
for _ , frame := range payloadFrames {
2018-02-17 00:29:53 -05:00
if err := frame . Write ( buffer , p . version ) ; err != nil {
2017-07-27 17:11:56 -05:00
return nil , err
}
}
2018-02-17 00:29:53 -05:00
// if this is an IETF QUIC Initial packet, we need to pad it to fulfill the minimum size requirement
// in gQUIC, padding is handled in the CHLO
if header . Type == protocol . PacketTypeInitial {
paddingLen := protocol . MinInitialPacketSize - sealer . Overhead ( ) - buffer . Len ( )
if paddingLen > 0 {
buffer . Write ( bytes . Repeat ( [ ] byte { 0 } , paddingLen ) )
}
}
if size := protocol . ByteCount ( buffer . Len ( ) + sealer . Overhead ( ) ) ; size > protocol . MaxPacketSize {
return nil , fmt . Errorf ( "PacketPacker BUG: packet too large (%d bytes, allowed %d bytes)" , size , protocol . MaxPacketSize )
2017-07-27 17:11:56 -05:00
}
raw = raw [ 0 : buffer . Len ( ) ]
2018-02-17 00:29:53 -05:00
_ = sealer . Seal ( raw [ payloadStartIndex : payloadStartIndex ] , raw [ payloadStartIndex : ] , header . PacketNumber , raw [ : payloadStartIndex ] )
raw = raw [ 0 : buffer . Len ( ) + sealer . Overhead ( ) ]
2017-07-27 17:11:56 -05:00
num := p . packetNumberGenerator . Pop ( )
2018-02-17 00:29:53 -05:00
if num != header . PacketNumber {
2017-07-27 17:11:56 -05:00
return nil , errors . New ( "packetPacker BUG: Peeked and Popped packet numbers do not match" )
}
2018-02-17 00:29:53 -05:00
p . hasSentPacket = true
2017-07-27 17:11:56 -05:00
return raw , nil
}
func ( p * packetPacker ) canSendData ( encLevel protocol . EncryptionLevel ) bool {
if p . perspective == protocol . PerspectiveClient {
return encLevel >= protocol . EncryptionSecure
}
return encLevel == protocol . EncryptionForwardSecure
2017-05-27 14:30:11 -05:00
}
2017-07-27 17:11:56 -05:00
func ( p * packetPacker ) SetLeastUnacked ( leastUnacked protocol . PacketNumber ) {
p . leastUnacked = leastUnacked
2017-05-27 14:30:11 -05:00
}
2018-02-17 00:29:53 -05:00
func ( p * packetPacker ) SetOmitConnectionID ( ) {
p . omitConnectionID = true
}