mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-05 13:17:36 +03:00
wire: optimize generation of Version Negotiation packets (#4278)
* wire: optimize generation of Version Negotiation packets * protocol: optimize adding greased version numbers
This commit is contained in:
parent
fbaa941ea1
commit
0582e931a5
3 changed files with 39 additions and 22 deletions
|
@ -1,10 +1,13 @@
|
||||||
package protocol
|
package protocol
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"crypto/rand"
|
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math"
|
"math"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"golang.org/x/exp/rand"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Version is a version number as int
|
// Version is a version number as int
|
||||||
|
@ -85,18 +88,24 @@ func ChooseSupportedVersion(ours, theirs []Version) (Version, bool) {
|
||||||
return 0, false
|
return 0, false
|
||||||
}
|
}
|
||||||
|
|
||||||
// generateReservedVersion generates a reserved version number (v & 0x0f0f0f0f == 0x0a0a0a0a)
|
var (
|
||||||
|
versionNegotiationMx sync.Mutex
|
||||||
|
versionNegotiationRand = rand.New(rand.NewSource(uint64(time.Now().UnixNano())))
|
||||||
|
)
|
||||||
|
|
||||||
|
// generateReservedVersion generates a reserved version (v & 0x0f0f0f0f == 0x0a0a0a0a)
|
||||||
func generateReservedVersion() Version {
|
func generateReservedVersion() Version {
|
||||||
b := make([]byte, 4)
|
var b [4]byte
|
||||||
_, _ = rand.Read(b) // ignore the error here. Failure to read random data doesn't break anything
|
_, _ = versionNegotiationRand.Read(b[:]) // ignore the error here. Failure to read random data doesn't break anything
|
||||||
return Version((binary.BigEndian.Uint32(b) | 0x0a0a0a0a) & 0xfafafafa)
|
return Version((binary.BigEndian.Uint32(b[:]) | 0x0a0a0a0a) & 0xfafafafa)
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetGreasedVersions adds one reserved version number to a slice of version numbers, at a random position
|
// GetGreasedVersions adds one reserved version number to a slice of version numbers, at a random position.
|
||||||
|
// It doesn't modify the supported slice.
|
||||||
func GetGreasedVersions(supported []Version) []Version {
|
func GetGreasedVersions(supported []Version) []Version {
|
||||||
b := make([]byte, 1)
|
versionNegotiationMx.Lock()
|
||||||
_, _ = rand.Read(b) // ignore the error here. Failure to read random data doesn't break anything
|
defer versionNegotiationMx.Unlock()
|
||||||
randPos := int(b[0]) % (len(supported) + 1)
|
randPos := rand.Intn(len(supported) + 1)
|
||||||
greased := make([]Version, len(supported)+1)
|
greased := make([]Version, len(supported)+1)
|
||||||
copy(greased, supported[:randPos])
|
copy(greased, supported[:randPos])
|
||||||
greased[randPos] = generateReservedVersion()
|
greased[randPos] = generateReservedVersion()
|
||||||
|
|
|
@ -1,13 +1,11 @@
|
||||||
package wire
|
package wire
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"crypto/rand"
|
"crypto/rand"
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"errors"
|
"errors"
|
||||||
|
|
||||||
"github.com/quic-go/quic-go/internal/protocol"
|
"github.com/quic-go/quic-go/internal/protocol"
|
||||||
"github.com/quic-go/quic-go/internal/utils"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// ParseVersionNegotiationPacket parses a Version Negotiation packet.
|
// ParseVersionNegotiationPacket parses a Version Negotiation packet.
|
||||||
|
@ -37,20 +35,19 @@ func ParseVersionNegotiationPacket(b []byte) (dest, src protocol.ArbitraryLenCon
|
||||||
func ComposeVersionNegotiation(destConnID, srcConnID protocol.ArbitraryLenConnectionID, versions []protocol.Version) []byte {
|
func ComposeVersionNegotiation(destConnID, srcConnID protocol.ArbitraryLenConnectionID, versions []protocol.Version) []byte {
|
||||||
greasedVersions := protocol.GetGreasedVersions(versions)
|
greasedVersions := protocol.GetGreasedVersions(versions)
|
||||||
expectedLen := 1 /* type byte */ + 4 /* version field */ + 1 /* dest connection ID length field */ + destConnID.Len() + 1 /* src connection ID length field */ + srcConnID.Len() + len(greasedVersions)*4
|
expectedLen := 1 /* type byte */ + 4 /* version field */ + 1 /* dest connection ID length field */ + destConnID.Len() + 1 /* src connection ID length field */ + srcConnID.Len() + len(greasedVersions)*4
|
||||||
buf := bytes.NewBuffer(make([]byte, 0, expectedLen))
|
buf := make([]byte, 1+4 /* type byte and version field */, expectedLen)
|
||||||
r := make([]byte, 1)
|
_, _ = rand.Read(buf[:1]) // ignore the error here. It is not critical to have perfect random here.
|
||||||
_, _ = rand.Read(r) // ignore the error here. It is not critical to have perfect random here.
|
|
||||||
// Setting the "QUIC bit" (0x40) is not required by the RFC,
|
// Setting the "QUIC bit" (0x40) is not required by the RFC,
|
||||||
// but it allows clients to demultiplex QUIC with a long list of other protocols.
|
// but it allows clients to demultiplex QUIC with a long list of other protocols.
|
||||||
// See RFC 9443 and https://mailarchive.ietf.org/arch/msg/quic/oR4kxGKY6mjtPC1CZegY1ED4beg/ for details.
|
// See RFC 9443 and https://mailarchive.ietf.org/arch/msg/quic/oR4kxGKY6mjtPC1CZegY1ED4beg/ for details.
|
||||||
buf.WriteByte(r[0] | 0xc0)
|
buf[0] |= 0xc0
|
||||||
utils.BigEndian.WriteUint32(buf, 0) // version 0
|
// The next 4 bytes are left at 0 (version number).
|
||||||
buf.WriteByte(uint8(destConnID.Len()))
|
buf = append(buf, uint8(destConnID.Len()))
|
||||||
buf.Write(destConnID.Bytes())
|
buf = append(buf, destConnID.Bytes()...)
|
||||||
buf.WriteByte(uint8(srcConnID.Len()))
|
buf = append(buf, uint8(srcConnID.Len()))
|
||||||
buf.Write(srcConnID.Bytes())
|
buf = append(buf, srcConnID.Bytes()...)
|
||||||
for _, v := range greasedVersions {
|
for _, v := range greasedVersions {
|
||||||
utils.BigEndian.WriteUint32(buf, uint32(v))
|
buf = binary.BigEndian.AppendUint32(buf, uint32(v))
|
||||||
}
|
}
|
||||||
return buf.Bytes()
|
return buf
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@ package wire
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
|
"testing"
|
||||||
|
|
||||||
"golang.org/x/exp/rand"
|
"golang.org/x/exp/rand"
|
||||||
|
|
||||||
|
@ -91,3 +92,13 @@ var _ = Describe("Version Negotiation Packets", func() {
|
||||||
Expect(reservedVersion&0x0f0f0f0f == 0x0a0a0a0a).To(BeTrue()) // check that it's a greased version number
|
Expect(reservedVersion&0x0f0f0f0f == 0x0a0a0a0a).To(BeTrue()) // check that it's a greased version number
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
func BenchmarkComposeVersionNegotiationPacket(b *testing.B) {
|
||||||
|
b.ReportAllocs()
|
||||||
|
supportedVersions := []protocol.Version{protocol.Version2, protocol.Version1, 0x1337}
|
||||||
|
destConnID := protocol.ArbitraryLenConnectionID{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 0xa, 0xb, 0xc, 0xd}
|
||||||
|
srcConnID := protocol.ArbitraryLenConnectionID{10, 9, 8, 7, 6, 5, 4, 3, 2, 1}
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
ComposeVersionNegotiation(destConnID, srcConnID, supportedVersions)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue