mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-04 12:47:36 +03:00
catch EPERM sendmsg errors for the very first packet on Linux (#4111)
This commit is contained in:
parent
262cf0a592
commit
b344940f06
6 changed files with 57 additions and 4 deletions
16
send_conn.go
16
send_conn.go
|
@ -28,6 +28,9 @@ type sconn struct {
|
||||||
packetInfoOOB []byte
|
packetInfoOOB []byte
|
||||||
// If GSO enabled, and we receive a GSO error for this remote address, GSO is disabled.
|
// If GSO enabled, and we receive a GSO error for this remote address, GSO is disabled.
|
||||||
gotGSOError bool
|
gotGSOError bool
|
||||||
|
// Used to catch the error sometimes returned by the first sendmsg call on Linux,
|
||||||
|
// see https://github.com/golang/go/issues/63322.
|
||||||
|
wroteFirstPacket bool
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ sendConn = &sconn{}
|
var _ sendConn = &sconn{}
|
||||||
|
@ -56,7 +59,7 @@ func newSendConn(c rawConn, remote net.Addr, info packetInfo, logger utils.Logge
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *sconn) Write(p []byte, gsoSize uint16, ecn protocol.ECN) error {
|
func (c *sconn) Write(p []byte, gsoSize uint16, ecn protocol.ECN) error {
|
||||||
_, err := c.WritePacket(p, c.remoteAddr, c.packetInfoOOB, gsoSize, ecn)
|
err := c.writePacket(p, c.remoteAddr, c.packetInfoOOB, gsoSize, ecn)
|
||||||
if err != nil && isGSOError(err) {
|
if err != nil && isGSOError(err) {
|
||||||
// disable GSO for future calls
|
// disable GSO for future calls
|
||||||
c.gotGSOError = true
|
c.gotGSOError = true
|
||||||
|
@ -69,7 +72,7 @@ func (c *sconn) Write(p []byte, gsoSize uint16, ecn protocol.ECN) error {
|
||||||
if l > int(gsoSize) {
|
if l > int(gsoSize) {
|
||||||
l = int(gsoSize)
|
l = int(gsoSize)
|
||||||
}
|
}
|
||||||
if _, err := c.WritePacket(p[:l], c.remoteAddr, c.packetInfoOOB, 0, ecn); err != nil {
|
if err := c.writePacket(p[:l], c.remoteAddr, c.packetInfoOOB, 0, ecn); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
p = p[l:]
|
p = p[l:]
|
||||||
|
@ -79,6 +82,15 @@ func (c *sconn) Write(p []byte, gsoSize uint16, ecn protocol.ECN) error {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *sconn) writePacket(p []byte, addr net.Addr, oob []byte, gsoSize uint16, ecn protocol.ECN) error {
|
||||||
|
_, err := c.WritePacket(p, addr, oob, gsoSize, ecn)
|
||||||
|
if err != nil && !c.wroteFirstPacket && isPermissionError(err) {
|
||||||
|
_, err = c.WritePacket(p, addr, oob, gsoSize, ecn)
|
||||||
|
}
|
||||||
|
c.wroteFirstPacket = true
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
func (c *sconn) capabilities() connCapabilities {
|
func (c *sconn) capabilities() connCapabilities {
|
||||||
capabilities := c.rawConn.capabilities()
|
capabilities := c.rawConn.capabilities()
|
||||||
if capabilities.GSO {
|
if capabilities.GSO {
|
||||||
|
|
|
@ -76,4 +76,27 @@ var _ = Describe("Connection (for sending packets)", func() {
|
||||||
Expect(c.capabilities().GSO).To(BeFalse())
|
Expect(c.capabilities().GSO).To(BeFalse())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if runtime.GOOS == "linux" {
|
||||||
|
It("doesn't fail if the very first sendmsg call fails", func() {
|
||||||
|
rawConn := NewMockRawConn(mockCtrl)
|
||||||
|
rawConn.EXPECT().LocalAddr()
|
||||||
|
rawConn.EXPECT().capabilities().AnyTimes()
|
||||||
|
c := newSendConn(rawConn, remoteAddr, packetInfo{}, utils.DefaultLogger)
|
||||||
|
gomock.InOrder(
|
||||||
|
rawConn.EXPECT().WritePacket([]byte("foobar"), remoteAddr, gomock.Any(), gomock.Any(), protocol.ECNCE).Return(0, errNotPermitted),
|
||||||
|
rawConn.EXPECT().WritePacket([]byte("foobar"), remoteAddr, gomock.Any(), uint16(0), protocol.ECNCE).Return(6, nil),
|
||||||
|
)
|
||||||
|
Expect(c.Write([]byte("foobar"), 0, protocol.ECNCE)).To(Succeed())
|
||||||
|
})
|
||||||
|
|
||||||
|
It("fails if the sendmsg calls fail multiple times", func() {
|
||||||
|
rawConn := NewMockRawConn(mockCtrl)
|
||||||
|
rawConn.EXPECT().LocalAddr()
|
||||||
|
rawConn.EXPECT().capabilities().AnyTimes()
|
||||||
|
c := newSendConn(rawConn, remoteAddr, packetInfo{}, utils.DefaultLogger)
|
||||||
|
rawConn.EXPECT().WritePacket([]byte("foobar"), remoteAddr, gomock.Any(), gomock.Any(), protocol.ECNCE).Return(0, errNotPermitted).Times(2)
|
||||||
|
Expect(c.Write([]byte("foobar"), 0, protocol.ECNCE)).To(MatchError(errNotPermitted))
|
||||||
|
})
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
|
@ -97,3 +97,14 @@ func isGSOError(err error) bool {
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// The first sendmsg call on a new UDP socket sometimes errors on Linux.
|
||||||
|
// It's not clear why this happens.
|
||||||
|
// See https://github.com/golang/go/issues/63322.
|
||||||
|
func isPermissionError(err error) bool {
|
||||||
|
var serr *os.SyscallError
|
||||||
|
if errors.As(err, &serr) {
|
||||||
|
return serr.Syscall == "sendmsg" && serr.Err == unix.EPERM
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
|
@ -13,7 +13,10 @@ import (
|
||||||
. "github.com/onsi/gomega"
|
. "github.com/onsi/gomega"
|
||||||
)
|
)
|
||||||
|
|
||||||
var errGSO = &os.SyscallError{Err: unix.EIO}
|
var (
|
||||||
|
errGSO = &os.SyscallError{Err: unix.EIO}
|
||||||
|
errNotPermitted = &os.SyscallError{Syscall: "sendmsg", Err: unix.EPERM}
|
||||||
|
)
|
||||||
|
|
||||||
var _ = Describe("forcing a change of send and receive buffer sizes", func() {
|
var _ = Describe("forcing a change of send and receive buffer sizes", func() {
|
||||||
It("forces a change of the receive buffer size", func() {
|
It("forces a change of the receive buffer size", func() {
|
||||||
|
|
|
@ -7,3 +7,4 @@ 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 }
|
||||||
|
func isPermissionError(err error) bool { return false }
|
||||||
|
|
|
@ -4,4 +4,7 @@ package quic
|
||||||
|
|
||||||
import "errors"
|
import "errors"
|
||||||
|
|
||||||
var errGSO = errors.New("fake GSO error")
|
var (
|
||||||
|
errGSO = errors.New("fake GSO error")
|
||||||
|
errNotPermitted = errors.New("fake not permitted error")
|
||||||
|
)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue