| // Copyright 2016 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. |
| |
| package net |
| |
| import ( |
| "syscall" |
| |
| "golang.org/x/net/lif" |
| ) |
| |
| // 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) { |
| lls, err := lif.Links(syscall.AF_UNSPEC, "") |
| if err != nil { |
| return nil, err |
| } |
| var ift []Interface |
| for _, ll := range lls { |
| if ifindex != 0 && ifindex != ll.Index { |
| continue |
| } |
| ifi := Interface{Index: ll.Index, MTU: ll.MTU, Name: ll.Name, Flags: linkFlags(ll.Flags)} |
| if len(ll.Addr) > 0 { |
| ifi.HardwareAddr = HardwareAddr(ll.Addr) |
| } |
| ift = append(ift, ifi) |
| } |
| return ift, nil |
| } |
| |
| const ( |
| sysIFF_UP = 0x1 |
| sysIFF_BROADCAST = 0x2 |
| sysIFF_DEBUG = 0x4 |
| sysIFF_LOOPBACK = 0x8 |
| sysIFF_POINTOPOINT = 0x10 |
| sysIFF_NOTRAILERS = 0x20 |
| sysIFF_RUNNING = 0x40 |
| sysIFF_NOARP = 0x80 |
| sysIFF_PROMISC = 0x100 |
| sysIFF_ALLMULTI = 0x200 |
| sysIFF_INTELLIGENT = 0x400 |
| sysIFF_MULTICAST = 0x800 |
| sysIFF_MULTI_BCAST = 0x1000 |
| sysIFF_UNNUMBERED = 0x2000 |
| sysIFF_PRIVATE = 0x8000 |
| ) |
| |
| func linkFlags(rawFlags int) Flags { |
| var f Flags |
| if rawFlags&sysIFF_UP != 0 { |
| f |= FlagUp |
| } |
| if rawFlags&sysIFF_BROADCAST != 0 { |
| f |= FlagBroadcast |
| } |
| if rawFlags&sysIFF_LOOPBACK != 0 { |
| f |= FlagLoopback |
| } |
| if rawFlags&sysIFF_POINTOPOINT != 0 { |
| f |= FlagPointToPoint |
| } |
| if rawFlags&sysIFF_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) { |
| var name string |
| if ifi != nil { |
| name = ifi.Name |
| } |
| as, err := lif.Addrs(syscall.AF_UNSPEC, name) |
| if err != nil { |
| return nil, err |
| } |
| var ifat []Addr |
| for _, a := range as { |
| var ip IP |
| var mask IPMask |
| switch a := a.(type) { |
| case *lif.Inet4Addr: |
| ip = IPv4(a.IP[0], a.IP[1], a.IP[2], a.IP[3]) |
| mask = CIDRMask(a.PrefixLen, 8*IPv4len) |
| case *lif.Inet6Addr: |
| ip = make(IP, IPv6len) |
| copy(ip, a.IP[:]) |
| mask = CIDRMask(a.PrefixLen, 8*IPv6len) |
| } |
| ifat = append(ifat, &IPNet{IP: ip, Mask: mask}) |
| } |
| return ifat, nil |
| } |
| |
| // interfaceMulticastAddrTable returns addresses for a specific |
| // interface. |
| func interfaceMulticastAddrTable(ifi *Interface) ([]Addr, error) { |
| return nil, nil |
| } |