Skip to content

Commit f6ea062

Browse files
ianlancetaylorgopherbot
authored andcommitted
internal/routebsd: fix parsing network address of length zero
This applies CL 646555 from the net repository to this copy. For #70528 Change-Id: Ib7e23accfa3f278392e7bdca6f8544b8f1395e7e Reviewed-on: https://go-review.googlesource.com/c/go/+/646676 Reviewed-by: Damien Neil <[email protected]> Auto-Submit: Ian Lance Taylor <[email protected]> TryBot-Bypass: Ian Lance Taylor <[email protected]> Reviewed-by: Ian Lance Taylor <[email protected]>
1 parent b7c9cdd commit f6ea062

File tree

2 files changed

+40
-36
lines changed

2 files changed

+40
-36
lines changed

src/internal/routebsd/address.go

+39-35
Original file line numberDiff line numberDiff line change
@@ -102,44 +102,52 @@ func (a *InetAddr) Family() int {
102102
// parseInetAddr parses b as an internet address for IPv4 or IPv6.
103103
func parseInetAddr(af int, b []byte) (Addr, error) {
104104
const (
105-
off4 = 4 // offset of in_addr
106-
off6 = 8 // offset of in6_addr
105+
off4 = 4 // offset of in_addr
106+
off6 = 8 // offset of in6_addr
107+
ipv4Len = 4 // length of IPv4 address in bytes
108+
ipv6Len = 16 // length of IPv6 address in bytes
107109
)
108110
switch af {
109111
case syscall.AF_INET:
110-
if len(b) < (off4+1) || len(b) < int(b[0]) || b[0] == 0 {
112+
if len(b) < (off4+1) || len(b) < int(b[0]) {
111113
return nil, errInvalidAddr
112114
}
113115
sockAddrLen := int(b[0])
114-
var ip [4]byte
115-
n := off4 + 4
116-
if sockAddrLen < n {
117-
n = sockAddrLen
116+
var ip [ipv4Len]byte
117+
if sockAddrLen != 0 {
118+
// Calculate how many bytes of the address to copy:
119+
// either full IPv4 length or the available length.
120+
n := off4 + ipv4Len
121+
if sockAddrLen < n {
122+
n = sockAddrLen
123+
}
124+
copy(ip[:], b[off4:n])
118125
}
119-
copy(ip[:], b[off4:n])
120126
a := &InetAddr{
121127
IP: netip.AddrFrom4(ip),
122128
}
123129
return a, nil
124130
case syscall.AF_INET6:
125-
if len(b) < (off6+1) || len(b) < int(b[0]) || b[0] == 0 {
131+
if len(b) < (off6+1) || len(b) < int(b[0]) {
126132
return nil, errInvalidAddr
127133
}
134+
var ip [ipv6Len]byte
128135
sockAddrLen := int(b[0])
129-
n := off6 + 16
130-
if sockAddrLen < n {
131-
n = sockAddrLen
132-
}
133-
var ip [16]byte
134-
copy(ip[:], b[off6:n])
135-
if ip[0] == 0xfe && ip[1]&0xc0 == 0x80 || ip[0] == 0xff && (ip[1]&0x0f == 0x01 || ip[1]&0x0f == 0x02) {
136-
// KAME based IPv6 protocol stack usually
137-
// embeds the interface index in the
138-
// interface-local or link-local address as
139-
// the kernel-internal form.
140-
id := int(bigEndian.Uint16(ip[2:4]))
141-
if id != 0 {
142-
ip[2], ip[3] = 0, 0
136+
if sockaddrLen != 0 {
137+
n := off6 + ipv6Len
138+
if sockAddrLen < n {
139+
n = sockAddrLen
140+
}
141+
copy(ip[:], b[off6:n])
142+
if ip[0] == 0xfe && ip[1]&0xc0 == 0x80 || ip[0] == 0xff && (ip[1]&0x0f == 0x01 || ip[1]&0x0f == 0x02) {
143+
// KAME based IPv6 protocol stack usually
144+
// embeds the interface index in the
145+
// interface-local or link-local address as
146+
// the kernel-internal form.
147+
id := int(bigEndian.Uint16(ip[2:4]))
148+
if id != 0 {
149+
ip[2], ip[3] = 0, 0
150+
}
143151
}
144152
}
145153
// The kernel can provide an integer zone ID.
@@ -197,11 +205,11 @@ func parseKernelInetAddr(af int, b []byte) (int, Addr, error) {
197205
switch {
198206
case b[0] == syscall.SizeofSockaddrInet6:
199207
a := &InetAddr{
200-
IP: netip.AddrFrom16([16]byte(b[off6:off6+16])),
208+
IP: netip.AddrFrom16([16]byte(b[off6 : off6+16])),
201209
}
202210
return int(b[0]), a, nil
203211
case af == syscall.AF_INET6:
204-
var ab[16]byte
212+
var ab [16]byte
205213
if l-1 < off6 {
206214
copy(ab[:], b[1:l])
207215
} else {
@@ -213,7 +221,7 @@ func parseKernelInetAddr(af int, b []byte) (int, Addr, error) {
213221
return int(b[0]), a, nil
214222
case b[0] == syscall.SizeofSockaddrInet4:
215223
a := &InetAddr{
216-
IP: netip.AddrFrom4([4]byte(b[off4:off4+4])),
224+
IP: netip.AddrFrom4([4]byte(b[off4 : off4+4])),
217225
}
218226
return int(b[0]), a, nil
219227
default: // an old fashion, AF_UNSPEC or unknown means AF_INET
@@ -251,16 +259,12 @@ func parseAddrs(attrs uint, b []byte) ([]Addr, error) {
251259
}
252260
b = b[l:]
253261
case syscall.AF_INET, syscall.AF_INET6:
254-
// #70528: if the sockaddrlen is 0, no address to parse inside,
255-
// skip over the record.
256-
if b[0] > 0 {
257-
af = int(b[1])
258-
a, err := parseInetAddr(af, b)
259-
if err != nil {
260-
return nil, err
261-
}
262-
as[i] = a
262+
af = int(b[1])
263+
a, err := parseInetAddr(af, b)
264+
if err != nil {
265+
return nil, err
263266
}
267+
as[i] = a
264268
l := roundup(int(b[0]))
265269
if len(b) < l {
266270
return nil, errMessageTooShort

src/internal/routebsd/address_darwin_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -108,7 +108,7 @@ var parseAddrsOnDarwinLittleEndianTests = []parseAddrsOnDarwinTest{
108108
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00, 0x00,
109109
},
110110
[]Addr{
111-
nil,
111+
&InetAddr{IP: netip.AddrFrom16([16]byte{})},
112112
&InetAddr{IP: netip.AddrFrom16([16]byte{0xfe, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xf2, 0x2f, 0x4b, 0xff, 0xfe, 0x09, 0x3b, 0xff})},
113113
&InetAddr{IP: netip.AddrFrom16([16]byte{0xff, 0xff, 0xff, 0xff, 0xff, 0xff})},
114114
nil,

0 commit comments

Comments
 (0)