lin
2025-06-05 ed3dd9d3e7519a82bb871d5eedb24a2fa0c91f47
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
// Copyright 2011 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
 
// +build darwin dragonfly freebsd netbsd openbsd
 
package net
 
import (
   "syscall"
 
   "internal/x/net/route"
)
 
// If the ifindex is zero, interfaceTable returns mappings of all
// network interfaces. Otherwise it returns a mapping of a specific
// interface.
func interfaceTable(ifindex int) ([]Interface, error) {
   msgs, err := interfaceMessages(ifindex)
   if err != nil {
       return nil, err
   }
   n := len(msgs)
   if ifindex != 0 {
       n = 1
   }
   ift := make([]Interface, n)
   n = 0
   for _, m := range msgs {
       switch m := m.(type) {
       case *route.InterfaceMessage:
           if ifindex != 0 && ifindex != m.Index {
               continue
           }
           ift[n].Index = m.Index
           ift[n].Name = m.Name
           ift[n].Flags = linkFlags(m.Flags)
           if sa, ok := m.Addrs[syscall.RTAX_IFP].(*route.LinkAddr); ok && len(sa.Addr) > 0 {
               ift[n].HardwareAddr = make([]byte, len(sa.Addr))
               copy(ift[n].HardwareAddr, sa.Addr)
           }
           for _, sys := range m.Sys() {
               if imx, ok := sys.(*route.InterfaceMetrics); ok {
                   ift[n].MTU = imx.MTU
                   break
               }
           }
           n++
           if ifindex == m.Index {
               return ift[:n], nil
           }
       }
   }
   return ift[:n], nil
}
 
func linkFlags(rawFlags int) Flags {
   var f Flags
   if rawFlags&syscall.IFF_UP != 0 {
       f |= FlagUp
   }
   if rawFlags&syscall.IFF_BROADCAST != 0 {
       f |= FlagBroadcast
   }
   if rawFlags&syscall.IFF_LOOPBACK != 0 {
       f |= FlagLoopback
   }
   if rawFlags&syscall.IFF_POINTOPOINT != 0 {
       f |= FlagPointToPoint
   }
   if rawFlags&syscall.IFF_MULTICAST != 0 {
       f |= FlagMulticast
   }
   return f
}
 
// If the ifi is nil, interfaceAddrTable returns addresses for all
// network interfaces. Otherwise it returns addresses for a specific
// interface.
func interfaceAddrTable(ifi *Interface) ([]Addr, error) {
   index := 0
   if ifi != nil {
       index = ifi.Index
   }
   msgs, err := interfaceMessages(index)
   if err != nil {
       return nil, err
   }
   ifat := make([]Addr, 0, len(msgs))
   for _, m := range msgs {
       switch m := m.(type) {
       case *route.InterfaceAddrMessage:
           if index != 0 && index != m.Index {
               continue
           }
           var mask IPMask
           switch sa := m.Addrs[syscall.RTAX_NETMASK].(type) {
           case *route.Inet4Addr:
               mask = IPv4Mask(sa.IP[0], sa.IP[1], sa.IP[2], sa.IP[3])
           case *route.Inet6Addr:
               mask = make(IPMask, IPv6len)
               copy(mask, sa.IP[:])
           }
           var ip IP
           switch sa := m.Addrs[syscall.RTAX_IFA].(type) {
           case *route.Inet4Addr:
               ip = IPv4(sa.IP[0], sa.IP[1], sa.IP[2], sa.IP[3])
           case *route.Inet6Addr:
               ip = make(IP, IPv6len)
               copy(ip, sa.IP[:])
           }
           if ip != nil && mask != nil { // NetBSD may contain route.LinkAddr
               ifat = append(ifat, &IPNet{IP: ip, Mask: mask})
           }
       }
   }
   return ifat, nil
}