github.com/metacubex/quic-go@v0.44.1-0.20240520163451-20b689a59136/internal/handshake/aead.go (about) 1 package handshake 2 3 import ( 4 "encoding/binary" 5 6 "github.com/metacubex/quic-go/internal/protocol" 7 "github.com/metacubex/quic-go/internal/utils" 8 ) 9 10 func createAEAD(suite *cipherSuite, trafficSecret []byte, v protocol.Version) *xorNonceAEAD { 11 keyLabel := hkdfLabelKeyV1 12 ivLabel := hkdfLabelIVV1 13 if v == protocol.Version2 { 14 keyLabel = hkdfLabelKeyV2 15 ivLabel = hkdfLabelIVV2 16 } 17 key := hkdfExpandLabel(suite.Hash, trafficSecret, []byte{}, keyLabel, suite.KeyLen) 18 iv := hkdfExpandLabel(suite.Hash, trafficSecret, []byte{}, ivLabel, suite.IVLen()) 19 return suite.AEAD(key, iv) 20 } 21 22 type longHeaderSealer struct { 23 aead *xorNonceAEAD 24 headerProtector headerProtector 25 nonceBuf [8]byte 26 } 27 28 var _ LongHeaderSealer = &longHeaderSealer{} 29 30 func newLongHeaderSealer(aead *xorNonceAEAD, headerProtector headerProtector) LongHeaderSealer { 31 if aead.NonceSize() != 8 { 32 panic("unexpected nonce size") 33 } 34 return &longHeaderSealer{ 35 aead: aead, 36 headerProtector: headerProtector, 37 } 38 } 39 40 func (s *longHeaderSealer) Seal(dst, src []byte, pn protocol.PacketNumber, ad []byte) []byte { 41 binary.BigEndian.PutUint64(s.nonceBuf[:], uint64(pn)) 42 return s.aead.Seal(dst, s.nonceBuf[:], src, ad) 43 } 44 45 func (s *longHeaderSealer) EncryptHeader(sample []byte, firstByte *byte, pnBytes []byte) { 46 s.headerProtector.EncryptHeader(sample, firstByte, pnBytes) 47 } 48 49 func (s *longHeaderSealer) Overhead() int { 50 return s.aead.Overhead() 51 } 52 53 type longHeaderOpener struct { 54 aead *xorNonceAEAD 55 headerProtector headerProtector 56 highestRcvdPN protocol.PacketNumber // highest packet number received (which could be successfully unprotected) 57 58 // use a single array to avoid allocations 59 nonceBuf [8]byte 60 } 61 62 var _ LongHeaderOpener = &longHeaderOpener{} 63 64 func newLongHeaderOpener(aead *xorNonceAEAD, headerProtector headerProtector) LongHeaderOpener { 65 if aead.NonceSize() != 8 { 66 panic("unexpected nonce size") 67 } 68 return &longHeaderOpener{ 69 aead: aead, 70 headerProtector: headerProtector, 71 } 72 } 73 74 func (o *longHeaderOpener) DecodePacketNumber(wirePN protocol.PacketNumber, wirePNLen protocol.PacketNumberLen) protocol.PacketNumber { 75 return protocol.DecodePacketNumber(wirePNLen, o.highestRcvdPN, wirePN) 76 } 77 78 func (o *longHeaderOpener) Open(dst, src []byte, pn protocol.PacketNumber, ad []byte) ([]byte, error) { 79 binary.BigEndian.PutUint64(o.nonceBuf[:], uint64(pn)) 80 dec, err := o.aead.Open(dst, o.nonceBuf[:], src, ad) 81 if err == nil { 82 o.highestRcvdPN = utils.Max(o.highestRcvdPN, pn) 83 } else { 84 err = ErrDecryptionFailed 85 } 86 return dec, err 87 } 88 89 func (o *longHeaderOpener) DecryptHeader(sample []byte, firstByte *byte, pnBytes []byte) { 90 o.headerProtector.DecryptHeader(sample, firstByte, pnBytes) 91 }