Skip to content

Commit

Permalink
net: store IPv4 returned from cgo resolver as 4-byte slice net.IP
Browse files Browse the repository at this point in the history
net.IP states that a 16-byte slice can still be an IPv4 address.
But after netip.Addr is introduced, it requires extra care to keep
it as an IPv4 address when converting it to a netip.Addr using
netip.AddrFromSlice.

To address this issue, let's change the cgo resolver to return
4-byte net.IP for IPv4. The change will save us 12 bytes too.

Please note that the go resolver already return IPv4 as 4-byte
slice.

The test TestResolverLookupIP has been modified to cover this
behavior. So no new test is added.

Fixes #53554.

Change-Id: I0dc2a59ad785c0c67a7bc22433105529f055997f
GitHub-Last-Rev: bd7bb2f
GitHub-Pull-Request: #53638
Reviewed-on: https://go-review.googlesource.com/c/go/+/415580
Auto-Submit: Damien Neil <[email protected]>
TryBot-Result: Gopher Robot <[email protected]>
Reviewed-by: Bryan Mills <[email protected]>
Reviewed-by: Damien Neil <[email protected]>
Run-TryBot: Ian Lance Taylor <[email protected]>
  • Loading branch information
ZekeLu authored and gopherbot committed Nov 2, 2022
1 parent 07a70bc commit c53390b
Show file tree
Hide file tree
Showing 4 changed files with 19 additions and 23 deletions.
9 changes: 0 additions & 9 deletions src/net/cgo_unix.go
Original file line number Diff line number Diff line change
Expand Up @@ -327,12 +327,3 @@ func cgoSockaddr(ip IP, zone string) (*_C_struct_sockaddr, _C_socklen_t) {
}
return nil, 0
}

func copyIP(x IP) IP {
if len(x) < 16 {
return x.To16()
}
y := make(IP, len(x))
copy(y, x)
return y
}
6 changes: 6 additions & 0 deletions src/net/ip.go
Original file line number Diff line number Diff line change
Expand Up @@ -757,3 +757,9 @@ func ParseCIDR(s string) (IP, *IPNet, error) {
m := CIDRMask(n, 8*iplen)
return ip, &IPNet{IP: ip.Mask(m), Mask: m}, nil
}

func copyIP(x IP) IP {
y := make(IP, len(x))
copy(y, x)
return y
}
23 changes: 11 additions & 12 deletions src/net/lookup_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import (
"context"
"fmt"
"internal/testenv"
"net/netip"
"reflect"
"runtime"
"sort"
Expand Down Expand Up @@ -1289,18 +1290,16 @@ func TestResolverLookupIP(t *testing.T) {
t.Fatalf("DefaultResolver.LookupIP(%q, %q): failed with unexpected error: %v", network, host, err)
}

var v4Addrs []IP
var v6Addrs []IP
var v4Addrs []netip.Addr
var v6Addrs []netip.Addr
for _, ip := range ips {
switch {
case ip.To4() != nil:
// We need to skip the test below because To16 will
// convent an IPv4 address to an IPv4-mapped IPv6
// address.
v4Addrs = append(v4Addrs, ip)
case ip.To16() != nil:
v6Addrs = append(v6Addrs, ip)
default:
if addr, ok := netip.AddrFromSlice(ip); ok {
if addr.Is4() {
v4Addrs = append(v4Addrs, addr)
} else {
v6Addrs = append(v6Addrs, addr)
}
} else {
t.Fatalf("IP=%q is neither IPv4 nor IPv6", ip)
}
}
Expand All @@ -1322,7 +1321,7 @@ func TestResolverLookupIP(t *testing.T) {
t.Errorf("DefaultResolver.LookupIP(%q, %q): unexpected IPv4 addresses: %v", network, host, v4Addrs)
}
if network == "ip4" && len(v6Addrs) > 0 {
t.Errorf("DefaultResolver.LookupIP(%q, %q): unexpected IPv6 addresses: %v", network, host, v6Addrs)
t.Errorf("DefaultResolver.LookupIP(%q, %q): unexpected IPv6 or IPv4-mapped IPv6 addresses: %v", network, host, v6Addrs)
}
})
}
Expand Down
4 changes: 2 additions & 2 deletions src/net/lookup_windows.go
Original file line number Diff line number Diff line change
Expand Up @@ -134,11 +134,11 @@ func (r *Resolver) lookupIP(ctx context.Context, network, name string) ([]IPAddr
switch result.Family {
case syscall.AF_INET:
a := (*syscall.RawSockaddrInet4)(addr).Addr
addrs = append(addrs, IPAddr{IP: IPv4(a[0], a[1], a[2], a[3])})
addrs = append(addrs, IPAddr{IP: copyIP(a[:])})
case syscall.AF_INET6:
a := (*syscall.RawSockaddrInet6)(addr).Addr
zone := zoneCache.name(int((*syscall.RawSockaddrInet6)(addr).Scope_id))
addrs = append(addrs, IPAddr{IP: IP{a[0], a[1], a[2], a[3], a[4], a[5], a[6], a[7], a[8], a[9], a[10], a[11], a[12], a[13], a[14], a[15]}, Zone: zone})
addrs = append(addrs, IPAddr{IP: copyIP(a[:]), Zone: zone})
default:
return nil, &DNSError{Err: syscall.EWINDOWS.Error(), Name: name}
}
Expand Down

0 comments on commit c53390b

Please sign in to comment.