Skip to content

Commit b0eb4d6

Browse files
committed
quic: compute pnum len from max ack received, not sent
QUIC packet numbers are truncated to include only the least significant bits of the packet number. The number of bits which must be retained is computed based on the largest packet number known to have been received by the peer. See RFC 9000, section 17.1. We were incorrectly using the largest packet number we have received *from* the peer. Oops. (Test infrastructure change: Include the header byte in the testPacket structure, so we can see how many bytes the packet number was encoded with. Ignore this byte when comparing packets.) Change-Id: Iec17c69f007f8b39d14d24b0ca216c6a0018ae22 Reviewed-on: https://go-review.googlesource.com/c/net/+/545575 Reviewed-by: Jonathan Amsterdam <jba@google.com> LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
1 parent b952594 commit b0eb4d6

File tree

5 files changed

+61
-9
lines changed

5 files changed

+61
-9
lines changed

internal/quic/conn_send.go

+3-3
Original file line numberDiff line numberDiff line change
@@ -60,7 +60,7 @@ func (c *Conn) maybeSend(now time.Time) (next time.Time) {
6060
pad := false
6161
var sentInitial *sentPacket
6262
if c.keysInitial.canWrite() {
63-
pnumMaxAcked := c.acks[initialSpace].largestSeen()
63+
pnumMaxAcked := c.loss.spaces[initialSpace].maxAcked
6464
pnum := c.loss.nextNumber(initialSpace)
6565
p := longPacket{
6666
ptype: packetTypeInitial,
@@ -93,7 +93,7 @@ func (c *Conn) maybeSend(now time.Time) (next time.Time) {
9393

9494
// Handshake packet.
9595
if c.keysHandshake.canWrite() {
96-
pnumMaxAcked := c.acks[handshakeSpace].largestSeen()
96+
pnumMaxAcked := c.loss.spaces[handshakeSpace].maxAcked
9797
pnum := c.loss.nextNumber(handshakeSpace)
9898
p := longPacket{
9999
ptype: packetTypeHandshake,
@@ -124,7 +124,7 @@ func (c *Conn) maybeSend(now time.Time) (next time.Time) {
124124

125125
// 1-RTT packet.
126126
if c.keysAppData.canWrite() {
127-
pnumMaxAcked := c.acks[appDataSpace].largestSeen()
127+
pnumMaxAcked := c.loss.spaces[appDataSpace].maxAcked
128128
pnum := c.loss.nextNumber(appDataSpace)
129129
c.w.start1RTTPacket(pnum, pnumMaxAcked, dstConnID)
130130
c.appendFrames(now, appDataSpace, pnum, limit)

internal/quic/conn_send_test.go

+43
Original file line numberDiff line numberDiff line change
@@ -38,3 +38,46 @@ func TestAckElicitingAck(t *testing.T) {
3838
}
3939
t.Errorf("after sending %v PINGs, got no ack-eliciting response", count)
4040
}
41+
42+
func TestSendPacketNumberSize(t *testing.T) {
43+
tc := newTestConn(t, clientSide, permissiveTransportParameters)
44+
tc.handshake()
45+
46+
recvPing := func() *testPacket {
47+
t.Helper()
48+
tc.conn.ping(appDataSpace)
49+
p := tc.readPacket()
50+
if p == nil {
51+
t.Fatalf("want packet containing PING, got none")
52+
}
53+
return p
54+
}
55+
56+
// Desynchronize the packet numbers the conn is sending and the ones it is receiving,
57+
// by having the conn send a number of unacked packets.
58+
for i := 0; i < 16; i++ {
59+
recvPing()
60+
}
61+
62+
// Establish the maximum packet number the conn has received an ACK for.
63+
maxAcked := recvPing().num
64+
tc.writeAckForAll()
65+
66+
// Make the conn send a sequence of packets.
67+
// Check that the packet number is encoded with two bytes once the difference between the
68+
// current packet and the max acked one is sufficiently large.
69+
for want := maxAcked + 1; want < maxAcked+0x100; want++ {
70+
p := recvPing()
71+
if p.num != want {
72+
t.Fatalf("received packet number %v, want %v", p.num, want)
73+
}
74+
gotPnumLen := int(p.header&0x03) + 1
75+
wantPnumLen := 1
76+
if p.num-maxAcked >= 0x80 {
77+
wantPnumLen = 2
78+
}
79+
if gotPnumLen != wantPnumLen {
80+
t.Fatalf("packet number 0x%x encoded with %v bytes, want %v (max acked = %v)", p.num, gotPnumLen, wantPnumLen, maxAcked)
81+
}
82+
}
83+
}

internal/quic/conn_test.go

+13-2
Original file line numberDiff line numberDiff line change
@@ -82,6 +82,7 @@ func (d testDatagram) String() string {
8282

8383
type testPacket struct {
8484
ptype packetType
85+
header byte
8586
version uint32
8687
num packetNumber
8788
keyPhaseBit bool
@@ -599,12 +600,18 @@ func (tc *testConn) readFrame() (debugFrame, packetType) {
599600
func (tc *testConn) wantDatagram(expectation string, want *testDatagram) {
600601
tc.t.Helper()
601602
got := tc.readDatagram()
602-
if !reflect.DeepEqual(got, want) {
603+
if !datagramEqual(got, want) {
603604
tc.t.Fatalf("%v:\ngot datagram: %v\nwant datagram: %v", expectation, got, want)
604605
}
605606
}
606607

607608
func datagramEqual(a, b *testDatagram) bool {
609+
if a == nil && b == nil {
610+
return true
611+
}
612+
if a == nil || b == nil {
613+
return false
614+
}
608615
if a.paddedSize != b.paddedSize ||
609616
a.addr != b.addr ||
610617
len(a.packets) != len(b.packets) {
@@ -622,16 +629,18 @@ func datagramEqual(a, b *testDatagram) bool {
622629
func (tc *testConn) wantPacket(expectation string, want *testPacket) {
623630
tc.t.Helper()
624631
got := tc.readPacket()
625-
if !reflect.DeepEqual(got, want) {
632+
if !packetEqual(got, want) {
626633
tc.t.Fatalf("%v:\ngot packet: %v\nwant packet: %v", expectation, got, want)
627634
}
628635
}
629636

630637
func packetEqual(a, b *testPacket) bool {
631638
ac := *a
632639
ac.frames = nil
640+
ac.header = 0
633641
bc := *b
634642
bc.frames = nil
643+
bc.header = 0
635644
if !reflect.DeepEqual(ac, bc) {
636645
return false
637646
}
@@ -839,6 +848,7 @@ func parseTestDatagram(t *testing.T, te *testEndpoint, tc *testConn, buf []byte)
839848
}
840849
d.packets = append(d.packets, &testPacket{
841850
ptype: p.ptype,
851+
header: buf[0],
842852
version: p.version,
843853
num: p.num,
844854
dstConnID: p.dstConnID,
@@ -880,6 +890,7 @@ func parseTestDatagram(t *testing.T, te *testEndpoint, tc *testConn, buf []byte)
880890
}
881891
d.packets = append(d.packets, &testPacket{
882892
ptype: packetType1RTT,
893+
header: hdr[0],
883894
num: pnum,
884895
dstConnID: hdr[1:][:len(tc.peerConnID)],
885896
keyPhaseBit: hdr[0]&keyPhaseBit != 0,

internal/quic/endpoint_test.go

+1-2
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,6 @@ import (
1313
"io"
1414
"net"
1515
"net/netip"
16-
"reflect"
1716
"testing"
1817
"time"
1918
)
@@ -242,7 +241,7 @@ func (te *testEndpoint) readDatagram() *testDatagram {
242241
func (te *testEndpoint) wantDatagram(expectation string, want *testDatagram) {
243242
te.t.Helper()
244243
got := te.readDatagram()
245-
if !reflect.DeepEqual(got, want) {
244+
if !datagramEqual(got, want) {
246245
te.t.Fatalf("%v:\ngot datagram: %v\nwant datagram: %v", expectation, got, want)
247246
}
248247
}

internal/quic/tls_test.go

+1-2
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,6 @@ import (
1010
"crypto/tls"
1111
"crypto/x509"
1212
"errors"
13-
"reflect"
1413
"testing"
1514
"time"
1615
)
@@ -56,7 +55,7 @@ func (tc *testConn) handshake() {
5655
fillCryptoFrames(want, tc.cryptoDataOut)
5756
i++
5857
}
59-
if !reflect.DeepEqual(got, want) {
58+
if !datagramEqual(got, want) {
6059
t.Fatalf("dgram %v:\ngot %v\n\nwant %v", i, got, want)
6160
}
6261
if i >= len(dgrams) {

0 commit comments

Comments
 (0)