1 // Copyright 2018 The Go Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style
3 // license that can be found in the LICENSE file.
9 "internal/syscall/unix"
14 type rawSockaddrDatalink struct {
30 const _KINFO_RT_IFLIST = (0x1 << 8) | 3 | (1 << 30)
32 const _RTAX_NETMASK = 2
36 const _SIOCGIFMTU = -0x3fd796aa
38 func getIfList() ([]byte, error) {
39 needed, err := syscall.Getkerninfo(_KINFO_RT_IFLIST, 0, 0, 0)
43 tab := make([]byte, needed)
44 _, err = syscall.Getkerninfo(_KINFO_RT_IFLIST, uintptr(unsafe.Pointer(&tab[0])), uintptr(unsafe.Pointer(&needed)), 0)
48 return tab[:needed], nil
51 // If the ifindex is zero, interfaceTable returns mappings of all
52 // network interfaces. Otherwise it returns a mapping of a specific
54 func interfaceTable(ifindex int) ([]Interface, error) {
55 tab, err := getIfList()
60 sock, err := sysSocket(syscall.AF_INET, syscall.SOCK_DGRAM, 0)
64 defer poll.CloseFunc(sock)
68 ifm := (*syscall.IfMsgHdr)(unsafe.Pointer(&tab[0]))
72 if ifm.Type == syscall.RTM_IFINFO {
73 if ifindex == 0 || ifindex == int(ifm.Index) {
74 sdl := (*rawSockaddrDatalink)(unsafe.Pointer(&tab[unsafe.Sizeof(syscall.IfMsgHdr)]))
76 ifi := &Interface{Index: int(ifm.Index), Flags: linkFlags(ifm.Flags)}
77 ifi.Name = string(sdl.Data[:sdl.Nlen])
78 ifi.HardwareAddr = sdl.Data[sdl.Nlen : sdl.Nlen+sdl.Alen]
82 copy(ifr.Name[:], ifi.Name)
83 err = unix.Ioctl(sock, syscall.SIOCGIFMTU, uintptr(unsafe.Pointer(ifr)))
87 ifi.MTU = int(ifr.Ifru[0])<<24 | int(ifr.Ifru[1])<<16 | int(ifr.Ifru[2])<<8 | int(ifr.Ifru[3])
89 ift = append(ift, *ifi)
90 if ifindex == int(ifm.Index) {
95 tab = tab[ifm.Msglen:]
101 func linkFlags(rawFlags int32) Flags {
103 if rawFlags&syscall.IFF_UP != 0 {
106 if rawFlags&syscall.IFF_BROADCAST != 0 {
109 if rawFlags&syscall.IFF_LOOPBACK != 0 {
112 if rawFlags&syscall.IFF_POINTOPOINT != 0 {
113 f |= FlagPointToPoint
115 if rawFlags&syscall.IFF_MULTICAST != 0 {
121 // If the ifi is nil, interfaceAddrTable returns addresses for all
122 // network interfaces. Otherwise it returns addresses for a specific
124 func interfaceAddrTable(ifi *Interface) ([]Addr, error) {
125 tab, err := getIfList()
132 ifm := (*syscall.IfMsgHdr)(unsafe.Pointer(&tab[0]))
136 if ifm.Type == syscall.RTM_NEWADDR {
137 if ifi == nil || ifi.Index == int(ifm.Index) {
139 off := uint(unsafe.Sizeof(syscall.IfMsgHdr))
141 var iprsa, nmrsa *syscall.RawSockaddr
142 for i := uint(0); i < _RTAX_MAX; i++ {
143 if mask&(1<<i) == 0 {
146 rsa := (*syscall.RawSockaddr)(unsafe.Pointer(&tab[off]))
147 if i == _RTAX_NETMASK {
153 off += (uint(rsa.Len) + 3) &^ 3
155 if iprsa != nil && nmrsa != nil {
159 switch iprsa.Family {
160 case syscall.AF_INET:
161 ipsa := (*syscall.RawSockaddrInet4)(unsafe.Pointer(iprsa))
162 nmsa := (*syscall.RawSockaddrInet4)(unsafe.Pointer(nmrsa))
163 ip = IPv4(ipsa.Addr[0], ipsa.Addr[1], ipsa.Addr[2], ipsa.Addr[3])
164 mask = IPv4Mask(nmsa.Addr[0], nmsa.Addr[1], nmsa.Addr[2], nmsa.Addr[3])
165 case syscall.AF_INET6:
166 ipsa := (*syscall.RawSockaddrInet6)(unsafe.Pointer(iprsa))
167 nmsa := (*syscall.RawSockaddrInet6)(unsafe.Pointer(nmrsa))
168 ip = make(IP, IPv6len)
169 copy(ip, ipsa.Addr[:])
170 mask = make(IPMask, IPv6len)
171 copy(mask, nmsa.Addr[:])
173 ifa := &IPNet{IP: ip, Mask: mask}
174 ifat = append(ifat, ifa)
178 tab = tab[ifm.Msglen:]
184 // interfaceMulticastAddrTable returns addresses for a specific
186 func interfaceMulticastAddrTable(ifi *Interface) ([]Addr, error) {