detect kernel GSO support

This commit is contained in:
Marten Seemann 2023-07-30 16:31:30 -04:00
parent 4122eb7a7d
commit 3a3169551b
7 changed files with 30 additions and 7 deletions

View file

@ -97,7 +97,9 @@ func (c *sconn) Write(p []byte, size protocol.ByteCount) error {
func (c *sconn) capabilities() connCapabilities { func (c *sconn) capabilities() connCapabilities {
capabilities := c.rawConn.capabilities() capabilities := c.rawConn.capabilities()
capabilities.GSO = !c.gotGSOError if capabilities.GSO {
capabilities.GSO = !c.gotGSOError
}
return capabilities return capabilities
} }

View file

@ -60,8 +60,8 @@ var _ = Describe("Connection (for sending packets)", func() {
Expect(oob).To(Equal(msg)) Expect(oob).To(Equal(msg))
return 0, errGSO return 0, errGSO
}), }),
rawConn.EXPECT().WritePacket([]byte("foo"), remoteAddr, []byte{}).Return(3, nil), rawConn.EXPECT().WritePacket([]byte("foo"), remoteAddr, gomock.Len(0)).Return(3, nil),
rawConn.EXPECT().WritePacket([]byte("bar"), remoteAddr, []byte{}).Return(3, nil), rawConn.EXPECT().WritePacket([]byte("bar"), remoteAddr, gomock.Len(0)).Return(3, nil),
) )
Expect(c.Write([]byte("foobar"), 3)).To(Succeed()) Expect(c.Write([]byte("foobar"), 3)).To(Succeed())
Expect(c.capabilities().GSO).To(BeFalse()) // GSO support is now disabled Expect(c.capabilities().GSO).To(BeFalse()) // GSO support is now disabled
@ -75,7 +75,7 @@ var _ = Describe("Connection (for sending packets)", func() {
rawConn.EXPECT().LocalAddr() rawConn.EXPECT().LocalAddr()
rawConn.EXPECT().capabilities() rawConn.EXPECT().capabilities()
c := newSendConn(rawConn, remoteAddr, packetInfo{}, utils.DefaultLogger) c := newSendConn(rawConn, remoteAddr, packetInfo{}, utils.DefaultLogger)
rawConn.EXPECT().WritePacket([]byte("foobar"), remoteAddr, nil) rawConn.EXPECT().WritePacket([]byte("foobar"), remoteAddr, gomock.Len(0))
Expect(c.Write([]byte("foobar"), 6)).To(Succeed()) Expect(c.Write([]byte("foobar"), 6)).To(Succeed())
}) })
} }

View file

@ -5,6 +5,7 @@ package quic
import ( import (
"encoding/binary" "encoding/binary"
"net/netip" "net/netip"
"syscall"
"golang.org/x/sys/unix" "golang.org/x/sys/unix"
) )
@ -29,3 +30,5 @@ func parseIPv4PktInfo(body []byte) (ip netip.Addr, ifIndex uint32, ok bool) {
} }
return netip.AddrFrom4(*(*[4]byte)(body[8:12])), binary.LittleEndian.Uint32(body), true return netip.AddrFrom4(*(*[4]byte)(body[8:12])), binary.LittleEndian.Uint32(body), true
} }
func isGSOSupported(syscall.RawConn) bool { return false }

View file

@ -4,6 +4,7 @@ package quic
import ( import (
"net/netip" "net/netip"
"syscall"
"golang.org/x/sys/unix" "golang.org/x/sys/unix"
) )
@ -24,3 +25,5 @@ func parseIPv4PktInfo(body []byte) (ip netip.Addr, _ uint32, ok bool) {
} }
return netip.AddrFrom4(*(*[4]byte)(body)), 0, true return netip.AddrFrom4(*(*[4]byte)(body)), 0, true
} }
func isGSOSupported(syscall.RawConn) bool { return false }

View file

@ -52,6 +52,18 @@ func parseIPv4PktInfo(body []byte) (ip netip.Addr, ifIndex uint32, ok bool) {
return netip.AddrFrom4(*(*[4]byte)(body[8:12])), binary.LittleEndian.Uint32(body), true return netip.AddrFrom4(*(*[4]byte)(body[8:12])), binary.LittleEndian.Uint32(body), true
} }
// isGSOSupported tests if the kernel supports GSO.
// Sending with GSO might still fail later on, if the interface doesn't support it (see isGSOError).
func isGSOSupported(conn syscall.RawConn) bool {
var serr error
if err := conn.Control(func(fd uintptr) {
_, serr = unix.GetsockoptInt(int(fd), unix.IPPROTO_UDP, unix.UDP_SEGMENT)
}); err != nil {
return false
}
return serr == nil
}
func appendUDPSegmentSizeMsg(b []byte, size uint16) []byte { func appendUDPSegmentSizeMsg(b []byte, size uint16) []byte {
startLen := len(b) startLen := len(b)
const dataLen = 2 // payload is a uint16 const dataLen = 2 // payload is a uint16

View file

@ -5,5 +5,5 @@ package quic
func forceSetReceiveBuffer(c any, bytes int) error { return nil } func forceSetReceiveBuffer(c any, bytes int) error { return nil }
func forceSetSendBuffer(c any, bytes int) error { return nil } func forceSetSendBuffer(c any, bytes int) error { return nil }
func appendUDPSegmentSizeMsg(_ []byte, _ uint16) []byte { return nil } func appendUDPSegmentSizeMsg([]byte, uint16) []byte { return nil }
func isGSOError(error) bool { return false } func isGSOError(error) bool { return false }

View file

@ -137,8 +137,11 @@ func newConn(c OOBCapablePacketConn, supportsDF bool) (*oobConn, error) {
batchConn: bc, batchConn: bc,
messages: msgs, messages: msgs,
readPos: batchSize, readPos: batchSize,
cap: connCapabilities{
DF: supportsDF,
GSO: isGSOSupported(rawConn),
},
} }
oobConn.cap.DF = supportsDF
for i := 0; i < batchSize; i++ { for i := 0; i < batchSize; i++ {
oobConn.messages[i].OOB = make([]byte, oobBufferSize) oobConn.messages[i].OOB = make([]byte, oobBufferSize)
} }