Skip to content

Commit cc4e616

Browse files
net: use mid-stack inlining with ReadFromUDP to avoid an allocation
This commit rewrites ReadFromUDP to be mid-stack inlined and pass a UDPAddr for lower layers to fill in. This lets performance-sensitive clients avoid an allocation. It requires some care on their part to prevent the UDPAddr from escaping, but it is now possible. The UDPAddr trivially does not escape in the benchmark, as it is immediately discarded. name old time/op new time/op delta WriteToReadFromUDP-8 17.2µs ± 6% 17.1µs ± 5% ~ (p=0.387 n=9+9) name old alloc/op new alloc/op delta WriteToReadFromUDP-8 112B ± 0% 64B ± 0% -42.86% (p=0.000 n=10+10) name old allocs/op new allocs/op delta WriteToReadFromUDP-8 3.00 ± 0% 2.00 ± 0% -33.33% (p=0.000 n=10+10) Updates #43451 Co-authored-by: Filippo Valsorda <[email protected]> Change-Id: I1f9d2ab66bd7e4eff07fe39000cfa0b45717bd13 Reviewed-on: https://go-review.googlesource.com/c/go/+/291509 Run-TryBot: Filippo Valsorda <[email protected]> TryBot-Result: Go Bot <[email protected]> Reviewed-by: Josh Bleecher Snyder <[email protected]> Reviewed-by: Jason A. Donenfeld <[email protected]> Trust: Filippo Valsorda <[email protected]> Trust: Josh Bleecher Snyder <[email protected]> Trust: Jason A. Donenfeld <[email protected]>
1 parent 2d4042d commit cc4e616

File tree

5 files changed

+45
-17
lines changed

5 files changed

+45
-17
lines changed

src/cmd/compile/internal/test/inl_test.go

+4-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ import (
1616
"testing"
1717
)
1818

19-
// TestIntendedInlining tests that specific runtime functions are inlined.
19+
// TestIntendedInlining tests that specific functions are inlined.
2020
// This allows refactoring for code clarity and re-use without fear that
2121
// changes to the compiler will cause silent performance regressions.
2222
func TestIntendedInlining(t *testing.T) {
@@ -155,6 +155,9 @@ func TestIntendedInlining(t *testing.T) {
155155
"(*rngSource).Int63",
156156
"(*rngSource).Uint64",
157157
},
158+
"net": {
159+
"(*UDPConn).ReadFromUDP",
160+
},
158161
}
159162

160163
if runtime.GOARCH != "386" && runtime.GOARCH != "mips64" && runtime.GOARCH != "mips64le" && runtime.GOARCH != "riscv64" {

src/net/udpsock.go

+13-9
Original file line numberDiff line numberDiff line change
@@ -100,11 +100,20 @@ func (c *UDPConn) SyscallConn() (syscall.RawConn, error) {
100100
}
101101

102102
// ReadFromUDP acts like ReadFrom but returns a UDPAddr.
103-
func (c *UDPConn) ReadFromUDP(b []byte) (int, *UDPAddr, error) {
103+
func (c *UDPConn) ReadFromUDP(b []byte) (n int, addr *UDPAddr, err error) {
104+
// This function is designed to allow the caller to control the lifetime
105+
// of the returned *UDPAddr and thereby prevent an allocation.
106+
// See https://blog.filippo.io/efficient-go-apis-with-the-inliner/.
107+
// The real work is done by readFromUDP, below.
108+
return c.readFromUDP(b, &UDPAddr{})
109+
}
110+
111+
// readFromUDP implements ReadFromUDP.
112+
func (c *UDPConn) readFromUDP(b []byte, addr *UDPAddr) (int, *UDPAddr, error) {
104113
if !c.ok() {
105114
return 0, nil, syscall.EINVAL
106115
}
107-
n, addr, err := c.readFrom(b)
116+
n, addr, err := c.readFrom(b, addr)
108117
if err != nil {
109118
err = &OpError{Op: "read", Net: c.fd.net, Source: c.fd.laddr, Addr: c.fd.raddr, Err: err}
110119
}
@@ -113,14 +122,9 @@ func (c *UDPConn) ReadFromUDP(b []byte) (int, *UDPAddr, error) {
113122

114123
// ReadFrom implements the PacketConn ReadFrom method.
115124
func (c *UDPConn) ReadFrom(b []byte) (int, Addr, error) {
116-
if !c.ok() {
117-
return 0, nil, syscall.EINVAL
118-
}
119-
n, addr, err := c.readFrom(b)
120-
if err != nil {
121-
err = &OpError{Op: "read", Net: c.fd.net, Source: c.fd.laddr, Addr: c.fd.raddr, Err: err}
122-
}
125+
n, addr, err := c.readFromUDP(b, &UDPAddr{})
123126
if addr == nil {
127+
// Return Addr(nil), not Addr(*UDPConn(nil)).
124128
return n, nil, err
125129
}
126130
return n, addr, err

src/net/udpsock_plan9.go

+4-3
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ import (
1111
"syscall"
1212
)
1313

14-
func (c *UDPConn) readFrom(b []byte) (n int, addr *UDPAddr, err error) {
14+
func (c *UDPConn) readFrom(b []byte, addr *UDPAddr) (int, *UDPAddr, error) {
1515
buf := make([]byte, udpHeaderSize+len(b))
1616
m, err := c.fd.Read(buf)
1717
if err != nil {
@@ -23,8 +23,9 @@ func (c *UDPConn) readFrom(b []byte) (n int, addr *UDPAddr, err error) {
2323
buf = buf[:m]
2424

2525
h, buf := unmarshalUDPHeader(buf)
26-
n = copy(b, buf)
27-
return n, &UDPAddr{IP: h.raddr, Port: int(h.rport)}, nil
26+
n := copy(b, buf)
27+
*addr = UDPAddr{IP: h.raddr, Port: int(h.rport)}
28+
return n, addr, nil
2829
}
2930

3031
func (c *UDPConn) readMsg(b, oob []byte) (n, oobn, flags int, addr *UDPAddr, err error) {

src/net/udpsock_posix.go

+3-4
Original file line numberDiff line numberDiff line change
@@ -43,14 +43,13 @@ func (a *UDPAddr) toLocal(net string) sockaddr {
4343
return &UDPAddr{loopbackIP(net), a.Port, a.Zone}
4444
}
4545

46-
func (c *UDPConn) readFrom(b []byte) (int, *UDPAddr, error) {
47-
var addr *UDPAddr
46+
func (c *UDPConn) readFrom(b []byte, addr *UDPAddr) (int, *UDPAddr, error) {
4847
n, sa, err := c.fd.readFrom(b)
4948
switch sa := sa.(type) {
5049
case *syscall.SockaddrInet4:
51-
addr = &UDPAddr{IP: sa.Addr[0:], Port: sa.Port}
50+
*addr = UDPAddr{IP: sa.Addr[0:], Port: sa.Port}
5251
case *syscall.SockaddrInet6:
53-
addr = &UDPAddr{IP: sa.Addr[0:], Port: sa.Port, Zone: zoneCache.name(int(sa.ZoneId))}
52+
*addr = UDPAddr{IP: sa.Addr[0:], Port: sa.Port, Zone: zoneCache.name(int(sa.ZoneId))}
5453
}
5554
return n, addr, err
5655
}

src/net/udpsock_test.go

+21
Original file line numberDiff line numberDiff line change
@@ -445,3 +445,24 @@ func TestUDPReadSizeError(t *testing.T) {
445445
}
446446
}
447447
}
448+
449+
func BenchmarkWriteToReadFromUDP(b *testing.B) {
450+
conn, err := ListenUDP("udp4", new(UDPAddr))
451+
if err != nil {
452+
b.Fatal(err)
453+
}
454+
addr := conn.LocalAddr()
455+
buf := make([]byte, 8)
456+
b.ResetTimer()
457+
b.ReportAllocs()
458+
for i := 0; i < b.N; i++ {
459+
_, err := conn.WriteTo(buf, addr)
460+
if err != nil {
461+
b.Fatal(err)
462+
}
463+
_, _, err = conn.ReadFromUDP(buf)
464+
if err != nil {
465+
b.Fatal(err)
466+
}
467+
}
468+
}

0 commit comments

Comments
 (0)