forked from quic-go/quic-go
add methods to change the remote address of the sendConn (#4933)
This commit is contained in:
50
send_conn.go
50
send_conn.go
@@ -2,6 +2,7 @@ package quic
|
||||
|
||||
import (
|
||||
"net"
|
||||
"sync/atomic"
|
||||
|
||||
"github.com/quic-go/quic-go/internal/protocol"
|
||||
"github.com/quic-go/quic-go/internal/utils"
|
||||
@@ -10,22 +11,29 @@ import (
|
||||
// A sendConn allows sending using a simple Write() on a non-connected packet conn.
|
||||
type sendConn interface {
|
||||
Write(b []byte, gsoSize uint16, ecn protocol.ECN) error
|
||||
WriteTo([]byte, net.Addr) error
|
||||
Close() error
|
||||
LocalAddr() net.Addr
|
||||
RemoteAddr() net.Addr
|
||||
ChangeRemoteAddr(addr net.Addr, info packetInfo)
|
||||
|
||||
capabilities() connCapabilities
|
||||
}
|
||||
|
||||
type remoteAddrInfo struct {
|
||||
addr net.Addr
|
||||
oob []byte
|
||||
}
|
||||
|
||||
type sconn struct {
|
||||
rawConn
|
||||
|
||||
localAddr net.Addr
|
||||
remoteAddr net.Addr
|
||||
localAddr net.Addr
|
||||
|
||||
remoteAddrInfo atomic.Pointer[remoteAddrInfo]
|
||||
|
||||
logger utils.Logger
|
||||
|
||||
packetInfoOOB []byte
|
||||
// If GSO enabled, and we receive a GSO error for this remote address, GSO is disabled.
|
||||
gotGSOError bool
|
||||
// Used to catch the error sometimes returned by the first sendmsg call on Linux,
|
||||
@@ -49,22 +57,26 @@ func newSendConn(c rawConn, remote net.Addr, info packetInfo, logger utils.Logge
|
||||
// increase oob slice capacity, so we can add the UDP_SEGMENT and ECN control messages without allocating
|
||||
l := len(oob)
|
||||
oob = append(oob, make([]byte, 64)...)[:l]
|
||||
return &sconn{
|
||||
rawConn: c,
|
||||
localAddr: localAddr,
|
||||
remoteAddr: remote,
|
||||
packetInfoOOB: oob,
|
||||
logger: logger,
|
||||
sc := &sconn{
|
||||
rawConn: c,
|
||||
localAddr: localAddr,
|
||||
logger: logger,
|
||||
}
|
||||
sc.remoteAddrInfo.Store(&remoteAddrInfo{
|
||||
addr: remote,
|
||||
oob: oob,
|
||||
})
|
||||
return sc
|
||||
}
|
||||
|
||||
func (c *sconn) Write(p []byte, gsoSize uint16, ecn protocol.ECN) error {
|
||||
err := c.writePacket(p, c.remoteAddr, c.packetInfoOOB, gsoSize, ecn)
|
||||
ai := c.remoteAddrInfo.Load()
|
||||
err := c.writePacket(p, ai.addr, ai.oob, gsoSize, ecn)
|
||||
if err != nil && isGSOError(err) {
|
||||
// disable GSO for future calls
|
||||
c.gotGSOError = true
|
||||
if c.logger.Debug() {
|
||||
c.logger.Debugf("GSO failed when sending to %s", c.remoteAddr)
|
||||
c.logger.Debugf("GSO failed when sending to %s", ai.addr)
|
||||
}
|
||||
// send out the packets one by one
|
||||
for len(p) > 0 {
|
||||
@@ -72,7 +84,7 @@ func (c *sconn) Write(p []byte, gsoSize uint16, ecn protocol.ECN) error {
|
||||
if 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], ai.addr, ai.oob, 0, ecn); err != nil {
|
||||
return err
|
||||
}
|
||||
p = p[l:]
|
||||
@@ -91,6 +103,11 @@ func (c *sconn) writePacket(p []byte, addr net.Addr, oob []byte, gsoSize uint16,
|
||||
return err
|
||||
}
|
||||
|
||||
func (c *sconn) WriteTo(b []byte, addr net.Addr) error {
|
||||
_, err := c.WritePacket(b, addr, nil, 0, protocol.ECNUnsupported)
|
||||
return err
|
||||
}
|
||||
|
||||
func (c *sconn) capabilities() connCapabilities {
|
||||
capabilities := c.rawConn.capabilities()
|
||||
if capabilities.GSO {
|
||||
@@ -99,5 +116,12 @@ func (c *sconn) capabilities() connCapabilities {
|
||||
return capabilities
|
||||
}
|
||||
|
||||
func (c *sconn) RemoteAddr() net.Addr { return c.remoteAddr }
|
||||
func (c *sconn) ChangeRemoteAddr(addr net.Addr, info packetInfo) {
|
||||
c.remoteAddrInfo.Store(&remoteAddrInfo{
|
||||
addr: addr,
|
||||
oob: info.OOB(),
|
||||
})
|
||||
}
|
||||
|
||||
func (c *sconn) RemoteAddr() net.Addr { return c.remoteAddrInfo.Load().addr }
|
||||
func (c *sconn) LocalAddr() net.Addr { return c.localAddr }
|
||||
|
||||
Reference in New Issue
Block a user