Skip to content

Commit 21814e7

Browse files
committedOct 3, 2023
quic: validate connection id transport parameters
Validate the original_destination_connection_id and initial_source_connection_id transport parameters. RFC 9000, Section 7.3 For golang/go#58547 Change-Id: I8343fd53c5cc946f15d3410c632b3895205fd597 Reviewed-on: https://go-review.googlesource.com/c/net/+/530036 Reviewed-by: Jonathan Amsterdam <jba@google.com> LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
1 parent a600b35 commit 21814e7

File tree

4 files changed

+87
-7
lines changed

4 files changed

+87
-7
lines changed
 

‎internal/quic/conn.go

+7-1
Original file line numberDiff line numberDiff line change
@@ -86,6 +86,7 @@ func newConn(now time.Time, side connSide, initialConnID []byte, peerAddr netip.
8686
// non-blocking operation.
8787
c.msgc = make(chan any, 1)
8888

89+
var originalDstConnID []byte
8990
if c.side == clientSide {
9091
if err := c.connIDState.initClient(c); err != nil {
9192
return nil, err
@@ -95,6 +96,7 @@ func newConn(now time.Time, side connSide, initialConnID []byte, peerAddr netip.
9596
if err := c.connIDState.initServer(c, initialConnID); err != nil {
9697
return nil, err
9798
}
99+
originalDstConnID = initialConnID
98100
}
99101

100102
// The smallest allowed maximum QUIC datagram size is 1200 bytes.
@@ -105,9 +107,10 @@ func newConn(now time.Time, side connSide, initialConnID []byte, peerAddr netip.
105107
c.streamsInit()
106108
c.lifetimeInit()
107109

108-
// TODO: initial_source_connection_id, retry_source_connection_id
110+
// TODO: retry_source_connection_id
109111
if err := c.startTLS(now, initialConnID, transportParameters{
110112
initialSrcConnID: c.connIDState.srcConnID(),
113+
originalDstConnID: originalDstConnID,
111114
ackDelayExponent: ackDelayExponent,
112115
maxUDPPayloadSize: maxUDPPayloadSize,
113116
maxAckDelay: maxAckDelay,
@@ -171,6 +174,9 @@ func (c *Conn) discardKeys(now time.Time, space numberSpace) {
171174

172175
// receiveTransportParameters applies transport parameters sent by the peer.
173176
func (c *Conn) receiveTransportParameters(p transportParameters) error {
177+
if err := c.connIDState.validateTransportParameters(c.side, p); err != nil {
178+
return err
179+
}
174180
c.streams.outflow.setMaxData(p.initialMaxData)
175181
c.streams.localLimit[bidiStream].setMax(p.initialMaxStreamsBidi)
176182
c.streams.localLimit[uniStream].setMax(p.initialMaxStreamsUni)

‎internal/quic/conn_id.go

+40-4
Original file line numberDiff line numberDiff line change
@@ -161,6 +161,39 @@ func (s *connIDState) issueLocalIDs(c *Conn) error {
161161
return nil
162162
}
163163

164+
// validateTransportParameters verifies the original_destination_connection_id and
165+
// initial_source_connection_id transport parameters match the expected values.
166+
func (s *connIDState) validateTransportParameters(side connSide, p transportParameters) error {
167+
// TODO: Consider returning more detailed errors, for debugging.
168+
switch side {
169+
case clientSide:
170+
// Verify original_destination_connection_id matches
171+
// the transient remote connection ID we chose.
172+
if len(s.remote) == 0 || s.remote[0].seq != -1 {
173+
return localTransportError(errInternal)
174+
}
175+
if !bytes.Equal(s.remote[0].cid, p.originalDstConnID) {
176+
return localTransportError(errTransportParameter)
177+
}
178+
// Remove the transient remote connection ID.
179+
// We have no further need for it.
180+
s.remote = append(s.remote[:0], s.remote[1:]...)
181+
case serverSide:
182+
if p.originalDstConnID != nil {
183+
// Clients do not send original_destination_connection_id.
184+
return localTransportError(errTransportParameter)
185+
}
186+
}
187+
// Verify initial_source_connection_id matches the first remote connection ID.
188+
if len(s.remote) == 0 || s.remote[0].seq != 0 {
189+
return localTransportError(errInternal)
190+
}
191+
if !bytes.Equal(p.initialSrcConnID, s.remote[0].cid) {
192+
return localTransportError(errTransportParameter)
193+
}
194+
return nil
195+
}
196+
164197
// handlePacket updates the connection ID state during the handshake
165198
// (Initial and Handshake packets).
166199
func (s *connIDState) handlePacket(c *Conn, ptype packetType, srcConnID []byte) {
@@ -170,10 +203,13 @@ func (s *connIDState) handlePacket(c *Conn, ptype packetType, srcConnID []byte)
170203
// We're a client connection processing the first Initial packet
171204
// from the server. Replace the transient remote connection ID
172205
// with the Source Connection ID from the packet.
173-
s.remote[0] = connID{
206+
// Leave the transient ID the list for now, since we'll need it when
207+
// processing the transport parameters.
208+
s.remote[0].retired = true
209+
s.remote = append(s.remote, connID{
174210
seq: 0,
175211
cid: cloneBytes(srcConnID),
176-
}
212+
})
177213
}
178214
case ptype == packetTypeInitial && c.side == serverSide:
179215
if len(s.remote) == 0 {
@@ -185,7 +221,7 @@ func (s *connIDState) handlePacket(c *Conn, ptype packetType, srcConnID []byte)
185221
})
186222
}
187223
case ptype == packetTypeHandshake && c.side == serverSide:
188-
if len(s.local) > 0 && s.local[0].seq == -1 {
224+
if len(s.local) > 0 && s.local[0].seq == -1 && !s.local[0].retired {
189225
// We're a server connection processing the first Handshake packet from
190226
// the client. Discard the transient, client-chosen connection ID used
191227
// for Initial packets; the client will never send it again.
@@ -213,7 +249,7 @@ func (s *connIDState) handleNewConnID(seq, retire int64, cid []byte, resetToken
213249
active := 0
214250
for i := range s.remote {
215251
rcid := &s.remote[i]
216-
if !rcid.retired && rcid.seq < s.retireRemotePriorTo {
252+
if !rcid.retired && rcid.seq >= 0 && rcid.seq < s.retireRemotePriorTo {
217253
s.retireRemote(rcid)
218254
}
219255
if !rcid.retired {

‎internal/quic/conn_id_test.go

+36-2
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,9 @@ func TestConnIDClientHandshake(t *testing.T) {
4848
t.Errorf("local ids: %v, want %v", fmtConnIDList(got), fmtConnIDList(wantLocal))
4949
}
5050
wantRemote := []connID{{
51+
cid: testLocalConnID(-1),
52+
seq: -1,
53+
}, {
5154
cid: testPeerConnID(0),
5255
seq: 0,
5356
}}
@@ -261,10 +264,12 @@ func TestConnIDPeerRetiresConnID(t *testing.T) {
261264
}
262265

263266
func TestConnIDPeerWithZeroLengthConnIDSendsNewConnectionID(t *testing.T) {
264-
// An endpoint that selects a zero-length connection ID during the handshake
267+
// "An endpoint that selects a zero-length connection ID during the handshake
265268
// cannot issue a new connection ID."
266269
// https://www.rfc-editor.org/rfc/rfc9000#section-5.1.1-8
267-
tc := newTestConn(t, clientSide)
270+
tc := newTestConn(t, clientSide, func(p *transportParameters) {
271+
p.initialSrcConnID = []byte{}
272+
})
268273
tc.peerConnID = []byte{}
269274
tc.ignoreFrame(frameTypeAck)
270275
tc.uncheckedHandshake()
@@ -536,6 +541,7 @@ func TestConnIDPeerWithZeroLengthIDProvidesPreferredAddr(t *testing.T) {
536541
// Peer gives us more conn ids than our advertised limit,
537542
// including a conn id in the preferred address transport parameter.
538543
tc := newTestConn(t, serverSide, func(p *transportParameters) {
544+
p.initialSrcConnID = []byte{}
539545
p.preferredAddrV4 = netip.MustParseAddrPort("0.0.0.0:0")
540546
p.preferredAddrV6 = netip.MustParseAddrPort("[::0]:0")
541547
p.preferredAddrConnID = testPeerConnID(1)
@@ -552,3 +558,31 @@ func TestConnIDPeerWithZeroLengthIDProvidesPreferredAddr(t *testing.T) {
552558
code: errProtocolViolation,
553559
})
554560
}
561+
562+
func TestConnIDInitialSrcConnIDMismatch(t *testing.T) {
563+
// "Endpoints MUST validate that received [initial_source_connection_id]
564+
// parameters match received connection ID values."
565+
// https://www.rfc-editor.org/rfc/rfc9000#section-7.3-3
566+
testSides(t, "", func(t *testing.T, side connSide) {
567+
tc := newTestConn(t, side, func(p *transportParameters) {
568+
p.initialSrcConnID = []byte("invalid")
569+
})
570+
tc.ignoreFrame(frameTypeAck)
571+
tc.ignoreFrame(frameTypeCrypto)
572+
tc.writeFrames(packetTypeInitial,
573+
debugFrameCrypto{
574+
data: tc.cryptoDataIn[tls.QUICEncryptionLevelInitial],
575+
})
576+
if side == clientSide {
577+
// Server transport parameters are carried in the Handshake packet.
578+
tc.writeFrames(packetTypeHandshake,
579+
debugFrameCrypto{
580+
data: tc.cryptoDataIn[tls.QUICEncryptionLevelHandshake],
581+
})
582+
}
583+
tc.wantFrame("initial_source_connection_id transport parameter mismatch",
584+
packetTypeInitial, debugFrameConnectionCloseTransport{
585+
code: errTransportParameter,
586+
})
587+
})
588+
}

‎internal/quic/conn_test.go

+4
Original file line numberDiff line numberDiff line change
@@ -201,6 +201,10 @@ func newTestConn(t *testing.T, side connSide, opts ...any) *testConn {
201201
TLSConfig: newTestTLSConfig(side),
202202
}
203203
peerProvidedParams := defaultTransportParameters()
204+
peerProvidedParams.initialSrcConnID = testPeerConnID(0)
205+
if side == clientSide {
206+
peerProvidedParams.originalDstConnID = testLocalConnID(-1)
207+
}
204208
for _, o := range opts {
205209
switch o := o.(type) {
206210
case func(*Config):

0 commit comments

Comments
 (0)
Please sign in to comment.